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数字 版 权 声 明 


图 灵 社 区 的 电子 书 没有 采用 专 有 客 
户 端 ， 您 可 以 在 任意 设备 上 ， 用 自 
己 喜 欢 的 浏览 器 和 PDF 阅读 器 进行 
阅读 。 

但 您 购买 的 电子 书 仅 供 您 个 人 使 
用 ,未 经 授权 ， 不 得 进行 传播 。 
我 们 愿意 相信 读者 具有 这 样 的 民 知 
和 觉悟 ， 与 我 们 共同 保护 知识 产 
权 。 

如 果 购 买 者 有 侵权 行为 ， 我 们 可 能 
对 该 用 户 实施 包括 但 不 限于 关闭 该 
帐号 等 维权 措施 ， 并 可 能 追究 法 律 


责任 。 





Gayle Laakmann McDowell 


美国 求职 咨询 网 站 CareerCup.com 创 始 人 兼 
CEO， 是 一 位 著名 软件 工程 师 ， 曾 在 微软 、 苹 果 
与 谷歌 任职 。 早 先 ， 她 自己 就 是 一 位 十 分 成 功 的 求 
职 者， 通过 了 微软 、 谷 歌 、 亚 马 钞 、 苹 果 、IBM、 高 
盛 等 多 家 最 著名 企业 极其 严 苛 的 面试 过 程 。 工 作 
以 后 ， 她 又 成 为 一 位 出 色 的 面试 官 。 在 谷歌 任职 
期 间 ， 她 还 是 该 公司 资深 面试 官 及 招聘 委员 会 成 
员 , 积累 了 相当 丰富 的 面试 经 验 。 除 此 书 外 , 她 
还 著 有 《人 金领 简历 : 敲 开 苹果 、 微 软 、 谷歌 的 大 
门 》 oo 


李 琳 戏 


主要 从 事 嵌 入 式 Linux 内 核 /驱动 开 发 ， 并 关 
注 IT、 开 放 源码 和 安防 监控 等 领域 。 业 余 时 间 
以 技术 翻译 为 乐 ， 时 而 客串 编辑 ， 好 为 爱 书 挑 
错 ， 渴 求 完美 ， 却 也 常 因 “小 ” 失 大 ， 不 得 读 
书 要 领 。 翻 译 或 参与 翻译 了 《Linux 命令 详解 
手册 》《 编 程 人 生 》《 编 程 大 师 访 谈 录 》 等 图 
书 。 网 络 ID 为 leal， 管 理 Vim、Android 等 
豆瓣 小 组 ， 个 人 站 点 : http://linxiao.net。 


漆 黎 

毕业 于 中 国 地 质 大 学 ， 拥 有 十 余年 软件 开发 、 测 
试 及 流程 管理 经 验 ， 热 囊 翻 译 ， 已 出 版 译作 包括 
《Linux/Unix 设 计 思 想 》《 人 金领 简历 : 敲 开 
苹果 、 微 软 、 谷 歌 的 大 门 》 等 书 。 目 前 定居 于 
美国 西雅图 , 在 微软 Windows Phone 开 发 中 心 
从 事 与 WP 应 用 开发 者 相关 的 项 目 管理 事务 。 
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内 容 提 要 


本 书 是 原 谷 歌 资深 面试 官 的 经 验 之 作 ， 层 层 紧 扣 程序 员 面 试 的 每 一 个 环节 ， 全 面 而 详尽 地 介绍 了 程序 
员 应 当 如 何 应 对 面试 ， 才 能 在 面试 中 脱颖而出 。 第 1 一 7 章 主要 涉及 面试 流程 解析 、 面 试 官 的 幕后 决策 及 可 
能 提出 的 问题 、 面 试 前 的 准备 工作 、 对 面试 结果 的 处 理 等 内 容 ; 第 8 一 9 章 从 数据 结构 、 概 念 与 算法 、 知 识 
类 问题 和 附加 面试 题 4 个 方面 ， 为 读者 呈现 了 出 自 微软 、 苹 果 、 谷 歌 等 多 家 知名 公司 的 150 道 编程 面试 题 ， 
并 针对 每 一 道 面 试题 目 ， 分 别 给 出 了 详细 的 解决 方案 。 

本 书 适合 程序 开发 和 设计 人 员 阅 读 。 
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亲爱 的 读者 : 

我 完 做 个 自我 介绍 。 

我 不 是 什么 招聘 人 员 。 我 只 是 一 名 软件 工程 师 。 正 因 如 此 , 我 深 知 大 家 要 在 面试 现场 迅速 想 
出 精妙 算法 并 在 白板 上 写 下 完美 代码 的 感受 。 之 所 以 能 感同身受 , 是 因为 我 与 你 们 有 过 同样 的 经 
历 ， 我 参加 过 谷歌 、 微 软 、 苹 果 、 亚 马 偿 以 及 其 他 诸多 公司 的 面试 。 

而 且 ， 我 也 当 过 面试 官 ， 让 求职 者 做 同样 的 事情 。 我 还 筛选 过 成 千 上 万 份 简历 , 在 其 中 “上 
下 求索 ”， 和 希望 挑 出 那些 或 许 能 在 面试 难关 中 脱颖而出 的 工程 师 。 在 谷歌 时 ， 我 与 招聘 委员 会 的 
同僚 有 过 激烈 争辩 ,探讨 某 位 求职 者 是 否 达到 了 录用 要 求 。 我 对 招聘 各 环节 了 如 指 掌 ， 相 关 经 验 
也 很 丰富 。 

而 现在 , 我 亲爱 的 读者 ,你 也 许 要 在 明天 、 下 周 或 是 明年 去 迎接 面试 挑战 。 你 可 能 已 经 拿 到 
或 者 正在 攻读 计算 机 科学 或 相关 专业 的 学 位 。 本 书 并 不 打算 给 大 家 重 温 有 关 二 又 查找 树 的 基本 知 
识 ,或 者 该 如 何 遍 历 链表 。 想 必 你 已 经 掌握 这 些 内 容 ; 倘若 没有 ,还 请 先 找 些 数据 结构 的 基础 资 
料 仔细 研读 。 

本 书 旨 在 帮助 你 加 深 对 计算 机 科学 基础 知识 的 理解 ,并 学 会 该 如 何 运用 这 些 基础 知识 , 成 功 
闻 过 技术 面试 这 一 关 。 

本 书 在 第 四 版 的 基础 上 做 了 大 量 更 新 , 增补 篇 幅 达 200 多 页 。 第 五 版 添补 了 不 少 面试 题 , 修 
订 了 部 分 原 有 题目 的 解法 , 并 新 增 了 几 个 章节 和 其 他 内 容 。 欢 迎 访问 我 们 的 网 站 , 你 可 以 跟 其 他 
求职 者 互通 有 无 ， 发 现 新 天 地 。 

与 此 同时 , 我 也 感到 无 比 兴 奋 , 你 一 定 能 从 本 书 中 学 到 新 的 技能 。 充 分 的 准备 会 让 你 在 技术 
和 人 际 沟通 技能 等 诸多 方面 更 进一步 。 不 管 最 终结 果 如 何 ， 只 要 拼 尽 全 力 ， 无 怨 无 悔 ! 

请 各 位 读者 务必 用 心 研 读本 书 前 面 的 介绍 性 章节 , 其 中 的 要 点 和 启示 也 许可 以 决定 你 的 面试 
结果 ,“ 录 用 ”与 “拒绝 ”就 在 一 线 之 间 。 
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此 外 ,切记 一 一 面试 非 易 事 ! 根据 我 在 谷歌 多 年 面试 的 经 历 , 我 留意 到 有 些 面试 官 会 问 一 些 
“简单 的 ”问题 ， 有 些 则 会 专 挑 难题 来 问 。 但 是 你 知道 吗 ? 面试 中 碰 到 简单 的 问题 ， 也 不 见得 就 
能 轻松 过 关 。 完 美 解决 问题 ( 只 有 极 少数 求职 者 才能 做 到 ! ) 不 是 公司 录用 你 的 关键 ， 只 有 题 答 
得 比 其 他 求职 者 更 出 色 才 能 让 你 脱颖而出 。 所 以 ,， 碰 到 杯 手 的 难题 也 不 要 惊慌 ,或许 其 他 人 一 样 
觉得 很 难 。 

请 努力 学 习 ， 不 断 实践 。 祝 你 好 运 ! 











盖 尔 ， 拉克 曼 . 麦克 道 尔 
CareerCup.com 创始 人 兼 CEO 
《金领 简历 : 圳 开 人 苹果、 微软、 谷歌 的 大 门 》 及 本 书 作 者 
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致谢 





生命 中 很 多 事情 都 离 不 开 团 队 合作 , 这 本 书 也 不 例外 。 在 创作 本 书 的 过 程 中 , 我 得 到 了 很 多 
人 的 帮助 ， 尽 管 涌 泉 都 难以 回报 ， 我 还 是 想 在 此 聊 表 十 心 。 

首先 , 我 要 感谢 我 的 丈夫 约翰 ,他 是 我 最 坚强 的 后 盾 ， 让 我 有 勇气 将 此 书 一 改 再 改 并 接连 修 
订 了 五 版 。 如 果 没 有 他 的 支持 ,我 很 可 能 就 做 不 到 这 一 点 。 

其 次 ,我 要 感谢 家 母 ， 她 让 我 认识 到 编程 无 比重 要 ， 而 写 出 优美 流畅 的 文字 更 为 重要 。 毫 无 
疑问 ， 她 是 一 位 无 与 伦比 的 工程 师 、 企 业 家 ， 最 重要 的 是 ， 她 是 一 位 伟大 的 母亲 。 

接 下 来 我 要 感谢 诸多 好 友 的 鼓励 ,尤其 是 加 尔 顿 英 格 里 绪 。 不 管 是 我 需要 帮助 还 是 听 我 发 
牢骚 ， 她 总 是 默默 陪 在 我 身边 ， 一 如 既往 地 支持 我 。 

最 后 , 我 还 要 感谢 那些 给 予 回 复 与 建议 的 读者 : 谢谢 你 们 ! 我 要 特别 感谢 维尼 特 ' 萨 哈 和 普 
拉 雷 … 瓦 玛 ,他 们 细致 入 微 地 审阅 了 书 中 的 每 一 道 题 , 用心 之 极 , 佩服 不 已 。 相 信 你 们 的 同事 和 
主管 一 定 会 为 有 这 么 出 色 的 伙伴 而 骄傲 。 

再 次 深 表 谢意 ! 
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招聘 中 的 问题 


讨论 完 招聘 事宜 ,我 们 又 一 次 诅 丧 地 走出 会 议 室 。 那 天 ,我 们 重新 审查 了 十 位 “过 关 ” 的 求 
职 者 ， 但 是 全 都 不 堪 录 用 。 我 们 很 纳 闽 ， 是 我 们 太 过 苟 刻 了 吗 ? 

我 尤为 失望 的 是 , 我 推荐 的 一 名 求职 者 也 被 拒 了 。 他 是 我 以 前 的 学 生 ， 以 高 达 3.73 的 平均 分 
(GPA ) 毕业 于 华盛顿 大 学 ， 这 可 是 世界 上 最 棒 的 计算 机 专业 院 校 之 一 。 此 外 ， 他 还 完成 了 大 量 
的 开源 项 目 工作 。 他 精力 充沛 、 富 于 创新 、 踏 实 能 干 、 头 脑 敏锐 ， 不 论 从 哪 方面 来 看 ， 他 都 堪 称 
真正 的 极 客 。 

但 是 , 我 不 得 不 同意 其 他 招聘 人 员 的 看 法 : 他 还 是 不 够 格 。 就 算 我 的 强力 推荐 可 以 让 他 侥幸 
过 关 ， 在 后 续 的 招聘 环节 还 是 会 失利 ， 因 为 他 的 硬 伤 太 多 了 。 

尽管 面试 官 都 认为 他 很 聪明 , 但 他 答题 总 是 奢 奢 绊 绊 的 。 大 多 数 成 功 的 求职 者 都 能 轻松 搞定 
第 一 道 题 ( 这 一 题 广为人知 ， 我 们 只 是 略 作 调整 而 已 )， 可 他 却 没 能 想 出 合适 的 算法 。 虽 然 他 后 
来 给 出 了 一 种 解法 , 但 没有 提出 针对 其 他 情形 进行 优化 的 解法 。 最 后 ,开始 写 代码 时 ,他 草草 地 
采用 了 最 初 的 思路 ， 可 这 个 解法 漏洞 百出 ,最 终 还 是 没 能 搞定 。 他 算 不 上 表现 最 差 的 求职 者 , 但 
与 我 们 的 “录用 底线 ” 却 相 去 其 远 ， 结 果 只 能 是 色 羽 而 归 。 

儿 个 星期 后 ,他 给 我 打 电 话 询问 反馈 意见 ,我 很 纠结 ,不 知 该 怎么 跟 他 说 。 他 需要 变 得 更 聪 
明 些 吗 ? 不 , 他 其 实 智力 超群 。 做 个 更 好 的 程序 员 ? 不 ,他 的 编程 技能 和 我 见 过 的 一 些 最 出 色 的 
程序 员 不 相 上 下 。 

跟 许多 积极 上 进 的 求职 者 一 样 , 他 准备 得 非常 充分 ,他 研读 过 Brian W. Kernighan 和 Dennis M. 
Ritchie 合 著 的 《C 程序 设计 语言 》 麻 省 理工 学 院 出 版 的 《算法 导论 》 等 经 典 著 作 。 他 可 以 细 数 
很 多 平衡 树 的 方法 ， 也 能 用 C 语 言 写 出 各 种 花哨 的 程序 。 

我 不 得 不 遗憾 地 告诉 他 : 光 是 看 这 些 书 还 远 远 不 够 。 这些 经 典 学 院 派 车 作 教 会 了 人 们 错 综 复 
杂 的 研究 理论 ， 对 程序 员 的 面试 却 助 益 不 多 。 为 什么 呢 ?” 容 我 稍稍 提醒 你 一 下 : 即使 从 学 生 时 代 
起 ， 你 的 面试 官 们 其 实 都 没 怎么 接触 过 所 谓 的 红 黑 树 ( Red-Black Trees ) 算法 。 

要 顺利 通过 面试 ， 就 得 “ 真 枪 实弹 ”地 做 准备 。 你 必须 演练 真正 的 面试 题 ， 并 掌握 它们 的 解 
题 模式 。 

这 本 书 就 是 我 根据 自己 在 顶尖 公司 积累 的 第 一 手 面试 经 验 提炼 而 成 的 精华 。 我 曾经 与 数 百名 
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求职 者 有 过 “交锋 "， 本 书 可 以 说 是 我 面试 几 百 位 求职 者 的 结晶 。 同 时 ， 我 还 从 成 千 上 万 求职 
与 面试 官 提 供 的 问题 中 精 挑 细 选 了 一 部 分 。 这 些 面试 题 出 自 许 多 知名 的 高 科技 公司 。 可 以 说 , 这 
本 书 赛 括 了 150 道 世 界 上 最 好 的 程序 员 面 试题 ， 都 是 从 数 以 千 计 的 好 问题 中 挑选 出 来 的 。 


我 的 写作 方法 


本 书 重 点 关注 算法 、 编 码 和 设计 问题 。 为 什么 呢 ? 尽管 面试 中 也 会 有 “行为 问题 ”， 但 是 答 
案 会 随 个 人 的 经 历 而 干 变 万 化。 同样， 尽管 许多 公司 也 会 考 问 细节 (例如 ,“ 什 么 是 虚 函 数 ? ”)， 
但 通过 演练 这 些 问题 而 取得 的 经 验 非常 有 限 , 更 多 地 是 涉及 非常 具体 的 知识 点 。 本 书 只 会 述 及 其 
中 一 些 问 题 ， 以 便 你 了 解 它们 “长 ”什么 样 。 当 然 ， 对 于 那些 可 以 拓展 技术 技能 的 问题 ， 我 会 给 
出 更 详细 的 解释 。 















































我 的 教学 热情 

我 特别 热爱 教学 。 我 喜欢 帮助 人 们 理解 新 概念 ， 并 提供 一 些 学 习 工具 ， 从 而 充分 激发 他 们 的 
学 习 热 情 。 

我 第 一 次 “正式 ”的 教学 经 验 是 在 美国 宾夕法尼亚 大 学 就 读 期 间 ， 那 时 我 才 大 二 ,担任 本 科 

















计算 机 科学 课程 的 助教 (TA )。 我 后 来 还 在 其 他 一 些 课程 中 担任 过 助教 ,最终 在 大 学 里 推出 了 自 
己 的 计算 机 科学 课程 ， 也 就 是 给 大 家 教授 一 些 实际 的 “动手 ”技能 。 

在 谷歌 担任 工程 师 时 , 培训 和 指导 “Nooglers”( 意 指 谷歌 新 员工 。 没 错 ， 他 们 就 是 这 么 称呼 
新 人 的 ! ) 是 我 最 喜欢 的 工作 之 一 。 后 来 ， 我 还 利用 “20% 自 由 支配 时 间 ” 在 华盛顿 大 学 教授 计 
算 机 科学 课程 。 

《程序 员 面 试 金 典 》《 金 领 简历 》 和 CareerCup.com 网 站 都 能 充分 体现 我 的 教学 热情 。 即 便 
是 现在 ， 你 也 会 发 现 我 经 常 出 现在 CareerCup.com 上 为 用 户 答疑 解 惑 。 

请 加 入 我 们 的 行列 吧 ! 





























Gayle Laakmann McDowell 
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第 1 章 


口 概述 

口 面试 题 的 来 源 

口 准备 时 间 表 与 注意 事项 
口 面试 评估 流程 
口 答题 情况 

口 着 装 规范 

口 十 大 常见 错误 
口 常见 问题 解答 


1.1 概述 


大 多 数 公司 的 面试 流程 其 实 都 大 同 小 异 。 本 章 会 简 述 面试 流程 ， 以 及 企业 到 底 想 招募 什 
么 样 的 人 才 。 这 些 信息 将 指导 你 如 何 做 好 面试 准备 ， 以 及 在 面试 过 程 中 和 面试 结束 后 该 如 何 
应 对 。 

收 到 面试 通知 后 ， 你 通常 得 先 经 历 一 次 筛选 面试 ( screening interview ), 一 般 通 过 电话 进行 。 
顶尖 高 校 的 应 届 毕 业 生 则 可 能 需要 参加 现场 的 筛选 面试 。 

不 要 因 “ 筛 选 面 坛 ”这 个 词 儿 而 掉以轻心 ， 筛 选 面试 也 很 有 可 能 涉及 编码 与 算法 问题 ， 要 求 
不 见得 比 现场 面试 低 。 如 果 不 确定 它 是 不 是 技术 筛选 面试 ,不 妨 问 问 招 聘 助 理 面试 官 是 什么 来 头 ， 
若是 工程 师 ， 那 十 有 八 九 会 与 技术 相关 。 

许多 公司 会 在 面试 中 运用 在 线 同步 文档 编辑 系统 , 但 也 有 可 能 让 你 直接 在 纸 上 写 好 代码 , 然 
后 在 电话 里 念 给 他 们 听 。 有 些 面 试 官 甚至 还 会 给 你 留 “ 家 庭 作 业 ”， 或 是 要 求 你 用 电子 邮件 将 写 
好 的 代码 发 给 他 们 。 

在 现场 面试 (on-site interview ) 之 前 ， 通 常会 有 一 两 轮 筛 选 面试 。 现 场面 试 大 概 有 4 到 6 轮 ， 
其 中 一 轮 可 能 是 午餐 面试 。 当 然 ， 午 餐 面 试 比较 随意 ,面试 官 一 般 不 会 问 你 技术 问题 ， 甚 至 不 会 
纳入 面试 评价 范畴 。 但 同时 ， 这 也 是 难得 的 好 机 会 ,你 可 以 跟 面 试 官 探讨 自己 感 兴趣 的 问题 ， 了 
解 公 司 的 企业 文化 。 其 他 几 轮 面试 主要 涉及 技术 问题 , 包括 编码 和 算法 等 。 此 外 ,你 可 能 还 要 回 
答 与 简历 相关 的 问题 。 
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2 第 1 章 面试 流程 





面试 结束 后 ， 面 试 官 们 会 聚 在 一 起 讨论 你 的 表现 ， 或 者 提交 书面 评价 。 大 多 数 情 况 下 ， 公 司 
招聘 人 员 都 会 在 一 周 内 给 你 回复 ， 告 知 应 聘 进 展 。 

要 是 已 经 望 穿 秋水 等 了 一 个 多 星期 ， 你 也 可 以 主动 询问 进展 。 就 算 招 聘 人 员 没 有 回应 ， 也 并 
不 表示 你 被 拒 了 ( 至少 大 的 高 科技 公司 是 这 样 ， 其 实 几乎 所 有 公司 都 是 如 此 )。 我 再 重复 一 次 : 
没有 回应 表示 你 的 应 聘 结果 还 是 未 知 数 。 当 然 , 人 们 都 希望 招聘 方 在 得 出 最 终结 论 时 ， 及 时 通知 
求职 者 。 

拖 拖 拉 拉 的 情况 确实 有 。 等 不 及 的 话 ， 不妨 问 问 相关 招聘 人 员 , 但 务 请 有 礼 有 节 。 招 聘 人 员 
和 我 们 一 样 ， 他 们 很 性 ， 有 些 人 会 因此 容易 忘 事 。 

































































1.2 面试 题 的 来 源 


求职 者 经 常会 问 我 ， 某 些 公 司 最 近 都 喜欢 问 哪些 面试 题 ? 他 们 总 以 为 面试 题 会 应 时 而 变 。 
实际 上 ， 公司 本 身 对 面试 题 并 没有 什么 倾向 ， 这 完全 取决 于 面试 官 的 个 人 喜好 。 容 我 解释 
= Fs 

在 大 公司 里 ,面试 官 通常 需要 先 参加 一 些 面试 培训 课程 。 在 谷歌 ,担任 面试 官 之 前 ,我 先 参 
加 了 一 次 由 外 部 公司 提供 的 专门 培训 。 培 训 课 程 为 期 一 天 ， 有 一 半 时 间 侧 重 于 法 律 层 面 的 事务 ， 
比如 ， 面 试 官 不 能 探 问 求 职 者 的 婚姻 状况 ,不 得 询问 种 族 ， 等 等 。 另 一 半 时 间 则 在 探讨 如 何 应 对 
“ 刺 头 ” 求 职 者 ， 比 如 当 问 及 编码 问题 或 其 他 令 求 职 者 认为 是 在 “ 关 厚 ”自己 的 问题 时 ， 要 是 求 
职 者 “ 暴 跳 如 雷 ”， 该 怎么 应 对 。 培 训 过 后 ， 我 又 实地 观摩 了 两 次 真正 的 面试 ， 然 后 就 开始 独自 
面试 了 。 

就 是 这 样 。 我 们 受过 的 培训 也 不 过 如 此 ， 甚 实 所 有 公司 都 大 同 小 异 。 

根本 就 不 存在 什么 “谷歌 官方 面试 题 清单 ”也 从 来 没有 人 要 求 我 一 定 要 问 哪些 特定 的 问题 ， 
或 者 必须 避 开 哪些 话题 。 

那 我 的 面试 题 从 何 而 来 呢 ? 其 实 ,来 源 和 大 家 一 样 。 

面试 官 也 当 过 求职 者 ,他 们 会 借用 自己 当年 被 拷问 过 的 题目 。 又 或 者 ， 有些 面试 官 也 会 彼此 
交换 题库 。 还 有 些 人 喜欢 上 网 找 问 题 ， 比 如 CareerCup.com 网 站 。 有 些 面试 官 也 可 能 从 上 述 渠道 
收集 面试 题 ， 并 或 多 或 少 做 些 调整 。 

就 算 真有 公司 给 面试 官 准 备 好 问题 清单 , 这 种 情况 也 并 不 多 见 。 面 试 官 通常 也 会 自行 挑选 问 
题 ， 而 且 大 家 往往 会 有 五 六 个 常用 的 备 选 题目 。 

因此 , 下 次 在 你 想 知道 谷歌 “最 近 ” 都 问 些 什么 问题 的 时 候 ， 不 妨 先 停 下 来 想 一 想 。 谷 歌 与 
亚马逊 的 面试 题 其 实 没什么 不 同 , 他 们 需要 的 都 是 软件 开发 人 才 。 至 于 面试 题 是 不 是 “最 近 流 行 
的 ”也 就 更 无 关 紧 要 了 了。 万 变 不 离 其 宗 ， 因 为 这 本 来 就 得 靠 面 试 官 自己 去 把 握 。 

当然 , 总 体 上 , 不 同 的 公司 在 风格 上 存在 差异 。 互联 网 公司 往往 会 提 些 系统 设计 方面 的 问题 ， 
而 那些 使 用 数据 库 的 公司 则 明显 偏爱 数据 库 方面 的 问题 。 然 而, 大 部 分 面试 题 无 外 乎 就 是 数据 结 
构 和 算法 之 类 的 ， 任 何 公司 都 会 问 到 。 
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1.3 ”准备 时 间 表 与 注意 事项 3 








1.3 准备 时 间 表 与 注意 事 Es 


Li 





“ 台 上 一 分 钟 ， 台 下 十 年 功 ”， 事 实 上 你 的 面试 表现 取决 于 你 的 功底 一 一 离 不 开 多 年 的 积淀 。 
你 需要 刚好 具备 能 为 公司 所 用 的 技术 经 验 , 然后 还 要 准备 好 在 面试 中 解决 实际 的 技术 问题 。 下 面 
的 时 间 表 和 流程 图 可 以 给 你 一 些 启发 。 

如 果 你 起 步 比 较 晚 ， 也 不 用 担心 。“ 尽 人 事 ， 知 天 命 "， 请 安心 准备 ， 视 你 好 运 ! 
























































































































































































































































1 年 bs 求学 /工作 之 。 | | 扩大 你 的 人 肪 
(面试 前 ) 余 做 些 项 目 及 社交 图 
y 
员 , 学 生 : 寻求 实习 a 本 
专注 重要 。。 | 二 | 机 会， 志 兴 有 大。 | < | 营建 加 站 (作品 入 
的 大 项 目 作业 的 课程 展示 自己 的 经 验 
继续 做 之 前 的 草拟 一 份 简历 ， 
人 | ”项 目 , 试 着 再 。 | 和 > | 。 找 个 过 来 人 审核 
加 个 项 目 一 下 
拆 一 份 清单 
玉 多 用 三 CT | 二 |。 阅读 本 书 前 几 章 。 | 如 | 。 列 出 自己 心仪 
人 的 公司 
y 
找 几 个 朋友 搭 做 些小 项 目 
档 ， 互相 进行 ”| > > 加 深 对 重要 
模拟 面试 概念 的 理解 
y 
列 一 份 清单 ， 
记录 解 题 时 < 继续 演练 面试 题 < 做 几 次 模拟 面试 
犯 过 的 错误 
y 
编制 一 份 面试 ey 
> 准备 表格 > 复审 /更 新 简 历 
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面试 流程 











重读 本 书 前 几 



























































































































































































































































































































































米 续 演 练 三 
章 ， 特 别 是 技 再 做 一 次 | 
术 和 行为 面试 模拟 面试 写 代码 
等 音节 
y 
电话 面试 找 个 现场 面试 : 建议 
舒服 地 儿 ， 买 个 将 面试 时 穿 的 套 
头 戴 式 耳机 装 送 去 干洗 
y 
,2 一 珀 一 页 演 面 试 准备 重 志 箔 六 昌 
进行 最 后 重读 算法 题 
次 模拟 面试 a 的 五 种 解法 
y 
rg 回顾 自己 re 
面试 前 一 天 We 继续 操练 面试 是 
y 
再 次 预演 面试 现场 面试 : 打印 继续 操练 面试 
备 表格 里 的 10 份 简历 ， 装 入 题 ， 并 回顾 
每 个 回答 文件 详 己 犯 过 的 错误 
y 
复习 2 的 宕 表 ， 重读 算法 题 的 
面试 当天 电话 面试 时 打 五 种 解法 ， 并 
印 一 份 牢记 在 心 
y 
晤 起 做 ey 
早餐 要 吃 好 ， E 
如 没 拿 到 offer 如 没收 到 招聘 A 中 
间 问 何 时 还 能 人 员 的 通知 ， 0 
应 聘 。 别 气 馆 ! 周 后 再 确认 DR 
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招聘 人 员 可 能 会 告诉 你 ,他 们 主要 考查 四 个 方面 : 工作 经 验 、 企 业 文 化 契合 度 、 编 程 技能 及 
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分 析 能 力 。 这 四 个 方面 相辅相成 , 但 在 决定 录用 与 否 时 ,分 量 最 重 的 通常 还 是 编程 技能 和 分 析 能 
力 (或 者 看 你 是 否 聪明 )。 这 也 是 为 什么 本 书 的 主要 篇 幅 都 在 探讨 如 何 提升 编程 与 算法 技能 。 
当然 ， 虽 说 编程 与 算法 技能 往往 最 为 重要 ， 但 并 不 表示 你 可 以 忽视 其 他 两 个 方面 。 

一 旦 进入 大 型 科技 公司 的 面试 环节 , 你 之 前 的 工作 经 验 就 不 是 特别 重要 了 , 但 它 可 能 会 左右 
面试 官 对 你 的 看 法 。 比 如 , 如 果 你 说 起 以 前 写 的 某 个 复杂 程序 的 精彩 之 处 , 面试 官 很 有 可 能 会 想 : 
“ 哇 ， 她 可 真 聪明 ! ”一旦 他 认定 你 智力 超群 ， 可 能 就 会 下 意识 地 忽略 你 所 犯 的 小 错误 。 总 之 , 面 
试 并 不 会 十 分 精确 ， 对 某 些 “ 软 问题 ”做 好 充分 准备 会 大 有 神 益 。 

创业 公司 比 大 公司 更 看 重 企业 文化 契合 度 (或 你 的 个 性 ， 主 要 看 是 否 与 公司 合拍 )。 举 个 例 
子 ， 如 果 公 司 的 企业 文化 鼓励 员工 独立 做 决定 ， 那 么 喜欢 听从 指导 的 人 就 不 太 适 合 了 。 

此 外 , 求职 者 因为 过 于 自 大 、 巧 辩 或 抵触 而 被 淘汰 的 情况 也 并 不 少见 。 我 就 遇 到 过 ， 有 位 求 
职 者 对 我 提问 的 用 词 吹 毛 求 疲 ,并 抱怨 这 导致 他 解 题 不 太 顺 利 , 后 来 他 还 对 我 的 引导 方式 心 生 不 
满 。 这 种 “抵触 心太 重 ” 的 表现 其 实 也 是 一 个 警示 ,果然 ,其 他 面试 官 对 他 的 感觉 也 很 不 好 。 最 
后 他 被 淘汰 了 。 谁 会 愿意 跟 这 种 人 一 起 共事 呢 ? 

所 以 ， 你 应 该 注意 以 下 几 点 。 

口 如 果 人 们 都 认为 你 骄傲 自 大 、 过 于 狭 辩 ， 或 有 其 他 负面 评价 ， 那 你 最 好 在 面试 中 收敛 一 
下 。 个 性 不 讨 喜 的 话 ， 哪 怕 你 的 表现 再 好 ， 也 可 能 会 被 拒 。 

口 准备 一 些 与 简历 相关 的 问题 。 虽 然 这 不 是 最 重要 的 因素 ， 但 也 不 能 掉以轻心 。 稍 微 花 点 
时 间 准 备 就 能 起 到 很 好 的 效果 ， 做 到 “四 两 拨 千 斤 ”。 

口 把 主要 精力 用 在 编程 与 算法 问题 上 。 

最 后 ,我 还 是 要 再 强调 一 遍 ， 面 试 并 不 会 十 分 精确 。 你 的 表现 可 能 会 有 失 水 准 ， 招 聘 委 员 会 
(或 不 管 是 谁 ) 有 时 候 也 会 做 出 错误 判断 。 就 像 任何 群体 一 样 ， 招 聘 委员 会 也 可 能 会 被 某 位 主导 
人 物 的 观点 所 左右 。 这 也 许 不 公平 ， 但 这 就 是 生活 。 

记 住 一 一 这 次 被 拒绝 并 不 代表 永远 。 一 年 内 你 还 可 以 重新 应 聘 , 很 多 求职 者 都 有 过 失利 后 再 
成 功 的 经 历 。 

不 要 气 馒 ， 失 败 是 成 功 之 母 。 


































































































































































































1.5 ”答题 情况 


有 则 谣传 流传 其 广 旦 颇具 迷惑 性 : 求职 者 必须 答对 全 部 问题 才 会 被 录用 。 事 实 绝 非 如 此 。 

首先 ， 面 试题 的 答案 很 难 用 “正确 ”和 “错误 ”去 简单 评判 。 我 个 人 在 评估 求职 者 的 面试 表 
现时 ,一般 不 会 只 看 他 们 答对 了 几 道 题 。 相 反 ， 我 会 考量 其 最 终 解 法 是 否 最 优 ， 用 时 多 久 , 代码 
整洁 与 否 。 这 不 只 是 单纯 的 是 非 判 断 ， 还 要 综合 考虑 很 多 因素 。 

其 次 , 你 的 面试 表现 还 会 拿 来 跟 其 他 求职 者 作 比 较 。 比 如 说 ,你 用 15 分 钟 出 色 地 解决 了 一 道 
题 ， 而 另 一 个 人 不 到 3 分钟 就 搞定 了 一 道 比 较 容 易 的 题 ， 是 否 就 意味 着 那个 人 的 表现 比 你 好 呢 ? 
也 许 是 ,但 也 未 必 。 很 自然 ， 面 试 官 出 的 题 越 简单 ， 他 们 越 是 希望 你 尽快 给 出 最 佳 答案 。 但 要 是 
题目 很 难 ， 他 们 也 不 会 指望 你 能 答 得 又 快 又 好 ， 毕 竞 ， 出 点 丝 漏 也 是 在 所 难免 的 。 
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我 在 谷歌 评估 过 数 千 名 求职 者 的 面试 资料 ， 其 中 只 有 一 位 求职 者 的 面试 表现 堪 称 “完美 无 
瑕 "。 其 他 人 ,包括 最 后 被 录用 的 几 百 个 幸运 儿 ， 都 或 多 或 少 犯 过 一 些 错 。 


1.6 着 装 规范 


软件 工程 师 一 般 都 穿 得 比较 随意 。 这 一 点 从 面试 的 着 装 规范 也 看 得 出 来 。 参 加 面试 时 ,推荐 
做 法 是 穿 得 比 同 级 别 员工 稍 好 一 点 。 

以 下 是 我 给 软件 工程 师 ( 及 测试 人 员 ) 的 面试 着 装 建议 ， 意 在 让 大 家 找到 一 个 “平衡 点 ”: 
不 要 穿 得 过 于 正式 ， 也 不 要 太 随 意 。 其 实 ， 有 很 多 人 还 是 穿着 牛仔 裤 和 T 恤 衫 参加 创业 公司 或 大 
公司 的 面试 ， 也 不 会 有 什么 问题 。 毕 竞 ， 公 司 不 是 看 你 穿 什么 ， 而 是 看 你 的 编程 水 平 。 







































































创业 公司 微软 、 谷 歌 、 亚 马 了 还 、Facebook 等 科技 巨头 | 非 科 技 公 司 〈 包 括 银行 ) 
卡其 裤 、 休 闲 裤 或 整洁 得 体 的 牛 | 卡其 裤 、 休 闲 裤 或 整洁 得 体 的 牛仔 裤 。Polo | 套装 ， 不 打 领 带 ( 可 带 
以 。Polo 衫 或 礼服 衬衫 衫 或 礼服 衬衫 一 条 领带 以 防 万 一 ) 
























































其 裤 、 休 闲 裤 或 整洁 得 体 的 牛 | 卡其 裤 、 休 闲 裤 或 整洁 得 体 的 牛仔 裤 。 大 方 | 套装 ， 或 得 体 的 休闲 裤 
裤 。 大 方 得 体 的 上 衣 或 毛衣 得 体 的 上 衣 或 毛衣 配 整 洁 的 上 衣 






































这 些 只 是 指导 建议 ， 具 体 还 要 参考 公司 的 企业 文化 。 此 外 ,如 果 你 应 聘 的 是 项 目 经 理 、 开 发 
主管 或 其 他 管理 层 职 位 ， 面 试 时 最 好 还 是 穿 得 正式 一 点 。 





1.7 十 大 常见 错误 


错误 一 : 只 在 计算 机 上 练习 

如 果 你 正 准 备 参 加 海洋 游泳 比赛 , 你 会 只 在 泳池 里 练习 吗 ? 应 该 不 会 。 你 得 去 体验 大 风 大 浪 
及 海洋 里 各 种 情况 带 来 的 影响 。 所 以 ， 你 肯定 会 希望 到 海洋 中 实地 训练 。 

在 计算 机 上 借助 编译 器 演练 面试 题 就 像 只 在 泳池 里 练习 一 样 。 抛 开 这 个 环境 吧 , 让 我 们 拿 出 
纸 和 笔 。 你 可 以 在 写 好 全 部 代码 并 做 过 人 工 测试 之 后 ， 再 在 计算 机 上 用 编译 器 进行 验证 。 

错误 二 : 不 做 行为 面试 题 演练 

很 多 求职 者 将 全 部 时 间 花 在 演练 技术 问题 上 ,而 忽视 了 行为 面试 题 。 你 猜 怎 么 着 ? 面试 官 可 
是 两 者 都 会 考查 的 。 

而 且 不 止 于 此 , 你 回答 行为 问题 的 表现 其 实 还 会 左右 面试 官 对 你 技术 能 力 的 看 法 。 行 为 问题 
的 准备 工作 其 实 相 对 比较 轻松 ， 而 且 容 易 达 到 事半功倍 的 效果 。 用 心 回顾 你 以 往 的 项 目 和 经 历 ， 
然后 准备 一 些小 故事 。 

错误 三 : 不 做 模拟 面试 训练 

假设 你 要 准备 一 场 重大 演讲 ， 所 有 同事 和 相关 人 员 都 将 列席 ,而 且 它 还 关乎 你 的 未 来 。 要 是 
只 在 头脑 里 无 声 地 练习 演讲 ， 到 了 真正 演讲 时 ， 你 肯定 会 发 狂 的 。 

光 是 纸上谈兵 ,不 做 模拟 面试 也 会 陷入 同样 的 境地 。 如 果 你 是 一 名 工程 师 ， 肯 定 认 识 不 少 同 
行 。 不 妨 找 个 朋友 帮 你 做 模拟 面试 。 作 为 回报 ， 你 也 可 以 给 他 当 一 回 面试 官 。 






























































图 灵 社区 会 员 cindy282694 专 享 尊重 版 权 


1.7 十 大 常见 错误 7 





着 误 四 : 试图 死记 硬 背 答案 

死记 人 硬 背 答案 最 多 只 能 解决 一 些 特定 问题 , 但 是 一 碰 到 新 的 题 ,你 可 能 就 傻眼 了 。 而 且 ， 基 
本 上 你 不 太 可 能 磁 上 出 自 本 书 的 题目 。 

最 靠 谱 的 做 法 就 是 , 不 看 答案 ， 先 把 书 里 的 题 全 部 认真 做 一 遍 。 这 样 你 才 有 可 能 练 就 各 种 技 
能 和 技巧 ,从 容 应 对 新 问题 。 就 算 最 后 你 只 能 大 概 复习 一 下 为 数 不 多 的 题 ， 这 种 做 法 也 会 对 你 很 
有 帮助 。 质 量 胜 于 数量 。 

普 误 五 : 不 大 声 说 出 你 的 解 题 思路 

透露 个 秘密 : 面试 官 才 不 会 知道 你 心里 想 什么 。 因 此 ,面试 时 黑 不 作 声 ,我 根本 无 法 了 解 你 
的 思路 。 假 如 你 沉默 时 间 过 长 ,我 还 会 误 以 为 你 毫 无 进展 。 你 得 多 出 声 ,， 没 准 说 着 说 着 就 找到 了 
解法 。 请 大 声 说 出 解 题 的 思路 ， 这 样 面试 官 就 会 知道 你 还 在 处 理 这 个 问题 ,没有 卡 壳 。 

这 么 做 还 有 个 好 处 就 是 不 至 于 跑题 ， 从 而 有 助 于 你 尽快 找到 解法 。 当 然 ， 最 大 的 作用 就 是 突 
显 你 强大 的 沟通 能 力 。 何 乐 而 不 为 呢 ? 

错误 六 : 过 于 仓促 

写 程序 不 是 什么 欧 赛 , 面试 也 不 是 , 所 以 解 题 时 不 要 太 过 仓促 。 代码 写 得 太 草 率 容易 出 问题 ， 
也 说 明 你 这 个 人 不 够 细心 。 请 放 慢 节奏 ， 有 条 不 麻 ， 多 做 测试 ， 问 题 考虑 得 周全 些 。 这 人 么 一 来 ， 
最 终 你 反而 能 更 高 效 地 给 出 答案 ， 错 误 也 会 少 一 些 。 

错误 七 : 代码 不 够 严谨 

其 实 每 个 人 都 写 得 出 完美 的 代码 ,但 有 时 我 们 还 是 会 在 面试 中 写 出 错误 百出 的 程序 ， 不 是 
吗 ? 代码 元 余 、 数 据 结构 乱七八糟 〈 比如 ， 缺少 面向 对 象 设计 ) 等 等 , 这些 都 是 常见 错误 ! 写 代 
码 时 , 不妨 设想 一 下 你 是 在 处 理 实际 问题 ， 要 注重 可 维护 性 。 将 代码 划分 成 不 同 的 子 程序 ， 并 精 
心 设计 数据 结构 来 处 理 相应 的 数据 。 

错误 八 : 不 做 测试 

在 日 常 工作 中 , 你 不 可 能 不 做 任何 测试 就 提交 代码 ,既然 如 此 ,为 什么 要 在 面试 中 省 略 这 一 
步 呢 ? 写 完 代 码 后 ， 请 “运行 ”( 或 者 审查 ) 一 下 程序 来 验证 结果 。 或 者 ， 在 处 理 复杂 问题 时 ， 
你 还 可 以 边 写 代码 边 测试 。 

错误 九 : 修正 错误 漫不经心 

程序 总 会 有 bug， 这 就 是 生活 或 编程 的 本 来 面目 。 只 要 用 心 测试 你 的 代码 ，bug 也 许 就 会 现 出 
原形 。 那 也 不 错 。 

不 过 ， 重 要 的 是 发 现 bug 时 ， 你 必须 三 思 而 后 行 ， 修 正之 前 先 确定 出 错 原因 。 有 些 求职 者 看 
到 传人 特定 参数 时 函数 返回 false 而 不 是 true， 会 直接 将 返回 值 取 反 ,接着 检查 问题 是 否 得 到 修 
正 。 当 然 ， 偶 尔 他们 也 能 频 猫 碰 上 死 耗子 ， 但 实际 上 如 此 仓促 行事 往往 会 导致 更 多 的 bug， 同 时 
也 反映 出 你 这 个 人 比较 粗心 大 意 。 

有 bug 其 实 很 正常 ， 但 胡乱 修改 代码 却 很 严重 。 

错误 十 : 轻 言 放弃 

我 知道 面试 题 都 很 难 , 但 不 难 怎么 显 出 求职 者 的 水 平 呢 。 你 会 迎 难 而 上 还 是 轻 言 放弃 ? 态度 
很 重要 ， 面 试 官 都 喜欢 那些 不 县 挑 成 、 迎 难 而 上 解决 问题 的 求职 者 。 毕 竟 ， 面 试 本 来 就 不 简单 。 














































































































图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 





8 第 1 章 面试 流程 








所 以 ， 碰 到 棘手 的 问题 请 不 要 惊慌 ， 也 不 要 轻 言 放弃 。 





1.8 常见 问题 解答 


1. 碰 到 熟悉 的 问题 时 应 该 如 实 相 告 吗 ? 

是 的 ! 碰 到 熟悉 的 问题 ， 当 然 要 告诉 面试 官 ! 有 些 人 会 觉得 这 很 俊 
( 并 知道 答案 )， 岂 不 是 如 虎 添 台 ， 对 吧 ? 其 实 ， 未 必 如 此 。 

我 们 力荐 你 如 实 相 告 的 理由 如 下 。 

(1) 彩 显 你 的 诚实 品质 。 这 能 反映 出 你 的 诚信 一 一 可 以 大 大 加 分 ! 要 知道 面试 官 可 是 在 默默 
地 考察 你 ,看 你 够 不 够 格 成 为 他 未 来 的 同事 。 我 不 知道 你 个 人 怎么 想 , 反正 我 是 喜欢 和 实在 人 一 
起 共事 。 

(2) 这 个 问题 可 能 略 有 改动 。 你 不 会 想 骨 这 个 险 给 个 错误 答案 吧 ? 

(3) 如 果 你 将 正确 答案 脱口 而 出 ， 面 试 官 会 觉得 很 可 疑 。 面 试 官 当然 知道 题目 的 难度 。 但 如 
果 你 伴 装 奢 奢 绊 绊 地 答题 ， 则 很 有 可 能 夸张 过 度 ， 而 显得 你 这 个 人 很 不 诚实 。 

2. 该 使 用 哪 种 编程 语言 ? 

很 多 人 都 会 建议 说 用 自己 最 得 心 应 手 的 语言 ,其实 理想 情况 下 , 你 应 该 使 用 面试 官 最 熟悉 的 
语言 。 我 一 般 会 推荐 使 用 C、C++ 或 Java， 因 为 大 多 数 面试 官 都 熟悉 这 三 种 语言 。 我 个 人 偏好 
Java ( 除非 涉及 C/C++ 问题 )， 因 为 用 Java 编 写 程序 效率 比较 高 ， 而 且 写 出 来 的 程序 简单 易 懂 ， 哪 
怕 平 时 用 惯 C++ 的 人 看 Java 程 序 也 不 会 有 太 大 难度 。 有 鉴于 此 ， 本 书 基本 上 都 用 Java 来 解 题 。 

3. 面试 结束 后 我 没有 收 到 回复 ， 是 被 拒 了 吗 ? 

不 是 的 。 真 要 被 拒 的 话 , 公司 一 般 都 会 给 你 通知 。 面 试 结束 后 短 时 间 内 没有 收 到 回复 并 不 代 
表 什 么 。 你 可 能 表现 得 很 不 错 , 但 招聘 人 员 不 巧 度假 去 了 , 没 能 及 时 处理。 公司 可 能 正在 进行 部 
门 重组 ,具体 该 招 多 少 人 尚 无 定论 。 叉 或 者 ,你 确实 表现 得 不 怎么 样 , 但 碰巧 遇 到 了 一 个 办 事 拖 
拉 或 者 特别 忙 的 招聘 人 员 ， 他 没 能 及 时 答复 你 。 当 然 ， 也 会 有 一 些 奇 怪 的 公司 。“ 嗯 ， 既 然 我 们 
不 打算 录用 这 个 求职 者 ， 那 就 没 必 要 给 他 回复 。” 所 以 ,一 切取 决 于 公司 本 身 。 但 你 可 以 发 邮件 
或 打 电 话 跟踪 后 续 进 展 。 

4. 被 拒 之 后 我 还 能 重新 申请 吗 ? 

当然 可 以 了 ， 不 过 通常 需要 等 上 一 段 时 间 (半年 至 一 年 )。 上 一 次 的 糟糕 表现 一 般 不 会 影响 
下 一 次 面试 。 很 多 人 都 被 微软 、 谷 歌 拒 过 ， 但 他 们 后 来 还 是 顺利 过 关 了 。 





要 是 熟悉 这 个 问题 
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口 微软 面试 

口 亚马逊 面试 
口 谷歌 面试 

口 苹果 面试 

口 Facebook 面 试 
口 雅虎 面试 














对 于 多 数 求职 者 而 言 , 面试 好 似 一 个 迷 局 。 你 去 了 , 见 了 几 个 面试 官 , 答 了 一 堆 问 题 , 然后 ， 
或 两 手 空空 离开 ， 或 幸运 地 拿 到 录用 通知 。 

你 有 没有 想 过 : 

口 面试 结果 是 怎么 得 出 的 ? 
口 面试 官 会 不 会 互相 交流 ? 
口 公司 最 看 重 哪些 方面 ? 

好 了 ,不 用 再 控 空 心思、 再 三 思索 了 ， 我 来 告诉 你 。 

在 本 章 ， 我们 邀请 了 来 自 顶 尖 科 技 公 司 (微软 、 亚 马 了 还、 谷歌、 苹果 、Facebook 及 雅虎 ) 的 
面试 专家 来 为 大 家 答疑 解 惑 ， 揭 秘 面试 中 的 那些 事 儿 。 

这 些 专家 会 让 我 们 了 解 各 家 公司 的 面试 流程 ， 帮 助 还 原 那 些 发 生 在 面试 会 议 室 之 外 的 事情 ， 
以 及 面试 结束 后 的 事项 。 

这 些 专 家 还 会 告诉 我 们 各 家 公司 面试 流程 的 不 同 之 处 。 比 如 ,亚马逊 的 “ 调 杆 员 ” “是 怎么 
回 事 ， 谷 歌 的 招聘 委员 会 是 如 何 运作 的 。 是 的 ， 每 家 公司 各 具 特 色 。 了 解 这 些 “ 怪 癖 ” 会 让 你 更 
加 胸 有 成 竹 ,不 会 被 突如其来 的 亚马逊 “ 调 杆 员 ” 给 吓 住 ， 也 不 会 对 苹果 居然 同时 铂 出 两 位 面试 
官 来 考察 你 而 感到 意外 。 

































































QD“barraiser”( 调 杆 员 ) 的 概念 来 自 亚马逊 美国 总 部 。 这 个 词 原 指 在 跳高 比赛 中 ， 一 次 次 将 杆 调 高 的 工作 人 员 。 而 
亚马逊 的 调 杆 员 则 是 一 群 在 招聘 过 程 中 负责 从 企业 文化 以 及 行为 准则 的 角度 考察 应 聘 者 ， 从 而 维护 招聘 质量 的 
人 。 在 招聘 中 ,， 调 杆 员 会 用 很 苛刻 的 眼光 考察 应 聘 者 是 否 在 至 少 一 点 上 高 过 亚马逊 的 平均 水 准 ， 如 果 是 ,那么 雇 
用 这 样 的 人 实际 上 就 等 于 在 提升 公司 的 能 力 ， 这 就 起 到 了 “ 抬 杆 ”的 作用 。 一 一 编者 注 
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此 外 , 这 些 专 家 也 强调 了 各 家 公司 的 面试 重点 。 尽 管 这 些 顶 尖 公 司 都 喜欢 考察 求职 者 的 编码 
能 力 和 算法 基础 ,他 们 其 实 也 各 有 侧重 。 不 管 这 是 源 自 各 家 公司 的 技术 背景 或 是 历史 ,至 少 你 知 
道 该 如 何 做 好 准备 。 

接 下 来 ， 让 我 们 一 起 揭 开 微软 、 亚 马 逊 、 谷 歌 、 苹 果 、Facebook 和 雅虎 的 “面试 迷 局 ” 吧 。 
2.1 微软 面试 

微软 喜欢 招 聪明 人 ,尤其 青睐 计算 机 极 客 。 求职 者 必须 对 技术 满怀 热情 。 微软 的 面试 官 不 大 
会 问 你 一 些 C++ API 的 个 中 细节 ， 而 是 直接 让 你 在 白板 上 写 代码 。 

参加 面试 时 , 求职 者 最 好 在 早上 约定 时 间 之 前 赶 到 微软 , 先 填 好 一 些 表格 。 接 着 你 会 和 招聘 
助理 碰面 ,他 会 给 你 一 个 面试 样题 。 招 聘 助理 主要 是 帮 你 热 热身 ,不 大 会 问 技术 问题 ; 就 算 真 的 
问 了 几 个 简单 的 技术 问题 ， 也 是 想 让 你 放松 心情 ， 等 到 面试 真正 开始 时 ， 你 就 不 会 那么 紧张 了 。 

对 招聘 助理 一 定 要 以 礼 相 待 。 说 不 定 他 们 会 玫 上 大 忙 , 在 你 首 轮 面试 表现 欠 佳 时 , 他 们 有 可 
能 帮 你 争取 重新 面试 的 机 会 。 夸 张 地 说 ， 他 们 甚至 还 能 左右 你 的 应 聘 结果 。 

面试 当天 你 会 接受 4 一 5 轮 面试 , 面试 官 一 般 来 
自 两 个 团队 。 许 多 公司 会 把 面试 安排 在 会 议 室 , 而 ” ”必要 准备 事项 































































































微软 的 面试 一 般 在 面试 官 的 办 公 室 进行 。 你 正好 可 “你 为 什么 想 要 加 入 微软 ? ” 
以 借 机 四 处 看 看 ， 感受 一 下 他 们 的 团队 文化 。 提 这 个 问题 ,微软 是 想 了 解 你 是 否 


一 轮 面试 过 后 , 不同 的 团队 ,做 法 不 一 样 , 面 。 对 技术 满怀 热情 。 一 个 比较 好 的 答案 
试 官 可 能 会 根据 个 人 习惯 决定 是 否 将 你 的 表现 反 ”是 “自打 接触 计算 机 以 来 ， 我 就 一 直 





僻 给 后 续 的 面试 。 在 用 微软 的 软件 ， 贵 公司 开发 的 软件 产 
完成 所 有 面试 后 ， 你 有 可 能 会 见 到 招聘 经 理 。 品 令 人 将 不 绝口 。 比 如 ， 我 最 近 一 直 在 








假如 真是 这 样 的 话 , 那 可 是 好 兆头 ， 这 意味 着 你 通 Visual Studio 开 发 环境 中 学 习 游 戏 编程 ， 

过 了 某 个 团队 的 基本 考察 。 接 下 来 ,就 要 看 招聘 经 。 它 的 API 实 在 是 太 好 用 了 。” 注 意 这 个 签 

理 要 不 要 录用 你 了 。 案 是 如 何 展示 你 对 技术 满怀 热情 的 。 
快 的 话 ， 面 试 当天 你 就 会 知道 结果 ， 慢 的 话 ， 二 二 

则 可 能 要 等 上 一 周 。 要 是 等 了 一 周 还 没收 到 人 事 部 | ee 

的 通知 ， 不 妨 发 封地 件 ， 客 气 地 问 一 下 进展 。 0 
如 果 你 没有 马上 收 到 回应 , 有 可 能 是 因为 招聘 肝 外 站 

助理 太 忙 了 ， 这 并 不 代表 你 就 没戏 。 

2.2 ”亚马逊 面试 

亚马逊 的 招聘 流程 一 般 从 两 轮 电话 面试 开始 , 期 间 求职 者 会 接受 某 个 团队 的 面试 。 偶 尔 也 会 

出 现 面试 3 轮 甚至 更 多 轮 的 情况 , 可 能 是 有 位 面试 官 对 你 的 评价 不 高 , 或 是 别 的 团队 对 你 有 兴 


此 外 , 还 有 其 他 特殊 情况 , 比如 求职 者 就 在 亚马逊 总 部 所 在 地 西雅图 , 或 他 以 前 面试 过 其 他 职位 ， 
也 许 一 次 电话 面试 就 够 了 。 
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2.3 ”谷歌 面试 ”11 








在 电话 面试 中 ， 面 试 你 的 工程 师 通常 会 要 求 你 通过 共享 文档 工具 〈 如 CollabEdit ) 写 些 简 














的 代码 。 他 们 问 的 技术 问题 可 谓 五 花 八 门 ， 意 在 探测 你 究竟 熟悉 哪些 领域 。 
接 下 来 , 如 有 一 两 个 团队 根据 你 的 简历 和 在 电话 面试 中 的 表现 相 中 你 , 你 就 要 飞 到 西雅图 
受 4 一 5 轮 面试 。 在 白板 上 写 代码 是 少不了 的 ,有些 面试 

















单 








官 会 着 重 考察 你 的 其 他 技能 。 每 一 轮 面 





接 
试 


官 都 会 侧重 不 同 的 领域 ,所 以 他 们 的 提问 会 大 相 径 庭 。 在 提交 自己 的 评价 报告 之 前 , 他 们 看 不 到 


其 他 面试 官 对 你 的 评价 , 而 且 公 司 也 不 鼓励 面试 官 
在 面试 过 程 中 互相 交流 , 一 切 讨论 都 得 等 到 几 轮 面 
试 全 部 结束 后 。 

顾名思义 ,“ 调 杆 员 ” 主 要 负责 把 控 面 试 质量 。 
他 们 受过 专门 训练 ， 并 且 是 从 其 他 团队 抽调 来 的 ， 
以 减少 面试 中 的 主观 倾向 。 在 面试 中 ,如 有 果 有 位 面 
试 官 风格 迎 异 上 且 要 求 格外 严格 , 那 他 可 能 就 是 传说 
中 的 “ 调 杆 员 "。 这 种 人 不 仅 面 试 经 验 丰富 ， 而 且 
跟 招 聘 经 理 一 样 ， 拥 有 生 杀 大 权 。 不 过 ,切记 : 这 
一 轮 面试 表现 磁石 绊 绊 ,并 不 等 于 你 的 整体 表现 就 
很 差 。 面 试 官 会 比照 其 他 求职 者 来 评价 你 的 水 平 ， 
而 不 是 只 看 你 答对 多 少 问题 。 

等 到 所 有 面试 官 提交 评价 报告 后 , 他 们 会 在 一 
起 讨论 你 的 表现 ， 并 决定 是 否 录用 你 。 

一 般 来 说 , 亚马逊 的 招聘 团队 都 会 很 快 给 出 录 
用 结果 ， 很 少 有 耽搁 。 要 是 一 周 内 都 没 等 到 结果 ， 
建议 你 发 封 措 辞 得 当 的 邮件 询问 进展 。 


谷歌 面试 


















































2.3 


必要 准备 事项 

亚马逊 是 一 家 互联 网 公司 ,这 也 意 
味 着 他 们 非常 关注 “扩展 性 ”问题 。 请 
做 好 相应 的 准备 。 当 然 , 回答 这 些 问 题 ， 
并 不 要 求 你 具备 分 布 式 系统 方面 的 知 
识 。 具 体 建议 可 参看 “扩展 性 与 存储 限 
六 

此 外 ,亚马逊 还 会 问 很 多 面向 对 象 
设计 的 问题 。 请 参看 “面向 对 象 设 计 ” 
一 节 ， 里 面 有 一 些 样题 和 建议 。 


独特 之 处 

“ 调 杆 员 ” 来 自 其 他 团队 ， 昌 在 提 
高 面试 标准 。 他 和 招聘 经 理 一 样 重要 ， 
请 尽量 表现 得 出 色 一 些 。 


业界 流传 着 很 多 有 关 谷 歌 面试 的 可 怕 谣 传 , 但 多 数 也 只 是 谣传 。 谷歌 的 面试 与 微软 或 亚马逊 


的 并 无 太 大 区 别 。 





谷歌 的 面试 也 从 电话 面试 开始 ,来 面试 你 的 人 是 技术 工程 师 ， 因 此 免不了 会 问 些 技术 难题 ， 
求职 者 切 不 可 掉以轻心 。 这 些 问 题 也 可 能 涉及 编程 ， 有 时 你 还 要 通过 共享 文档 工具 写 些 代码 。 电 


话 面试 的 问题 和 现场 面试 的 类 似 ， 要 求 也 一 样 。 


























现场 面试 一 般 有 4 一 6 轮 ， 其 中 一 轮 为 午餐 面试 。 面 试 官 之 间 不 能 透露 自己 的 评价 报告 ， 因 此 
每 一 轮 面试 你 都 可 以 从 零 开 始 。 午餐 面试 不 会 有 评价 报告 , 你 可 以 借 机 问 些 其 他 环节 不 方便 问 的 





问题 。 








谷歌 不 会 要 求 面 试 官 侧重 不 同 的 领域 , 也 没有 所 谓 的 标准 流程 或 结构 。 每 个 面试 官 可 以 自行 


决定 问 哪些 问题 。 


面试 过 后 ， 评 价 报告 会 以 书面 形式 提交 给 由 工程 师 和 经 理 组 成 的 “招聘 委员 会 ”, 
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由 他 们 


作 





12 第 2 章 面试 揭秘 





出 录用 结论 。 面试 评价 报告 由 分 析 能 力 、 编 程 水 平 、 
工作 经 验 和 沟通 能 力 等 四 部 分 组 成 ,最 后 你 会 得 到 
总 的 评分 ,在 1.0 到 4.0 之 间 。 "招聘 委员 会 ”里 一 般 
不 会 有 你 的 面试 官 。 就 算 有 ， 那 也 纯 属 巧合 。 

通常 , 在 决定 录用 与 否 时 , 招聘 委员 会 更 看 本 
那 种 有 面试 官 给 你 打 高 分 的 情况 ， 打 个 比方 ， 如 
果 你 的 得 分 是 3.6、3.1、3.1 和 2.6, 效果 要 好 过 拿 4 
个 3.1。 

也 就 是 说 ， 每 轮 面试 不 一 定 都 要 有 上 佳 表现 。 
此 外 ， 你 在 电话 面试 中 的 表现 一 般 起 不 了 决定 性 
作用 。 

如 果 招 聘 委员 会 给 出 的 意见 是 “聘用 ”， 你 的 
材料 就 会 转 给 “薪酬 委员 会 ”及 “执行 管理 委员 会 ”。 
最 终结 果 可 能 要 等 上 几 周 ， 因 为 还 有 不 少 流程 要 
走 ， 等 待 多 个 委员 会 审批 。 


2.4 ”苹果 面试 
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必要 准备 事项 

作为 一 家 互联 网 公司 , 谷歌 非常 看 
重 如 何 设计 可 扩展 的 系统 。 因此, 务必 
掌握 “扩展 性 与 存储 限制 ”一 节 的 问题 。 
此 外 ,谷歌 的 面试 官 很 喜欢 问 些 涉及 


“位 操作 ”的 问题 ， 也 请 重点 复习 这 些 
方面 的 知识 。 
独特 之 处 


面试 官 不 是 决策 者 。 他 们 只 提交 评 
价 意见 供 招 聘 委员 会 参考 。 招聘 委员 会 
给 出 录用 与 否 的 决定 ,当然 , 该 决定 偶 
尔 也 会 被 谷歌 高 管 否决 。 


苹果 的 面试 流程 与 公司 本 身 的 风格 非常 相符 , 是 最 没 官僚 味 儿 的 。 苹果 的 面试 官 很 看 重 技术 
功底 ,但 求职 者 对 应 聘 职位 和 公司 的 热情 也 非常 重要 。 虽 然 成 为 Mac 用 户 并 不 是 应 聘 苹 果 的 先决 





条 件 ， 但 你 至 少 要 对 该 系统 有 一 定 了 解 。 

在 苹果 的 面试 流程 中 , 招聘 助理 会 先 给 你 打 电 
话 了 解 一 些 基 本 情况 , 接 下 来 团队 成 员 会 对 你 进行 
一 连 串 的 技术 电话 面试 。 

当 你 受 邀 去 参加 现场 面试 时 , 招聘 助理 会 出 面 
接待 你 ， 并 介绍 面试 的 大 致 流程 。 然 后 ， 你 要 接受 
招聘 团队 成 员 6 一 8 轮 的 面试 , 其 中 这 个 团队 的 重要 
人 物 也 会 来 面试 你 。 

苹果 的 面试 形式 是 一 对 一 或 二 对 一 。 请 做 好 在 
白板 上 写 代 码 的 准备 , 交流 的 时 候 一 定 要 把 自己 的 
思路 表达 清楚 。 你 可 能 会 跟 未 来 的 上 司 共 进 午餐 ， 
这 看 似 随意 , 但 其 实 也 是 一 次 面试 。 每 个 面试 官 都 
会 侧重 不 同 的 领域 , 面试 官 之 间 一 般 不 会 过 问 彼此 
的 面试 情况 , 除非 他 们 想 让 后 续 面 试 官 就 求职 者 菏 
一 方面 多 挖掘 点 内 容 。 

当天 所 有 面试 结束 后 ,面试 官 会 在 一 起 商议 你 
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必要 准备 事项 

如 果 你 知道 哪个 团队 会 来 面试 你 ， 
务必 先 熟悉 他 们 的 产品 。 你 喜欢 该 产品 
的 哪些 方面 ? 你 觉得 有 哪些 可 以 改进 
的 地 方 ? 给 出 独到 见解 可 以 有 力 展示 
你 对 这 份 工 作 的 激情 。 


独特 之 处 

在 苹果 的 面试 中 , 二 对 一 的 形式 司 
室 见 惯 , 不 过 也 不 用 太 紧 张 一 一 这 跟 一 
对 一 面试 并 无 分 别 。 

此 外 ,苹果 的 员工 都 是 超级 果 粉 ， 
在 面试 中 ， 你 最 好 也 能 展现 出 同样 的 
热情 。 


2.$ ”Facebook 面试 13 





的 表现 。 如 果 大 家 都 认为 你 表现 不 错 , 接 下 来 会 由 你 应 聘 部 门 的 主管 或 副 总 来 面试 你 。 能 见 到 主 
管 也 不 见得 你 一 定 会 被 录用 , 不 过 总 归 是 个 好 兆头 。 让 不 让 你 见 主管 的 决定 对 你 是 不 公开 的 ,如 
果 你 落选 了 ， 他 们 只 是 默默 送 你 离开 公司 ， 也 不 会 透露 你 为 什么 落选 了 。 

如 果 你 得 以 进入 主管 或 副 总 面试 环节 , 面 过 你 的 面试 官 会 聚 到 会 议 室 正式 表决 录用 意见 。 副 
总 通常 不 会 列席 , 但 如 果 你 没 能 打动 他 们 ,他 们 照样 可 以 直接 否决 。 招聘 人 员 通 常会 在 几 天 后 联 
系 你 ， 要 是 等 不 及 的 话 ， 你 也 可 以 主动 联系 。 


2.5 Facebook 面试 


Facebook 的 在 线 工程 难题 " 曾 引发 热 议 , 其 实 这 无 非 又 是 吸引 眼球 的 手段 之 一 。 除 了 解答 这 
些 难题 ， 你 还 可 以 通过 传统 渠道 申请 该 公司 的 职位 ， 比 如 提交 在 线 职 位 申请 ,或 者 参加 校园 招 
聘 会 。 

一 旦 被 Facebook 挑 中 ,求职 者 一 般 至 少 要 接受 两 轮 电话 面试 。 不 过 ,公司 所 在 地 ”的 求职 者 可 
以 少 一 轮 。 电 话 面试 主要 涉及 技术 问题 ， 求 职 者 通常 要 用 Etherpad 或 其 他 共享 文档 工具 写 些 代码 。 

如 果 你 还 在 上 学 ， 在 学 校 接受 面试 ， 那 你 还 要 写 代 码 。 面 试 官 会 要 求 你 在 白板 或 日 纸 上 写 
代 人 码 0 

现场 面试 时 ， 主 要 由 其 他 软件 工程 师 来 面试 你 , 不 过 ,招聘 经 理 有 空 的 话 也 会 参与 。 所 有 面 
试 官 都 受过 专业 面试 培训 ， 他 们 只 提供 意见 ， 对 你 的 应 聘 结果 不 作 决 断 。 

现场 面试 的 每 个 面试 官 都 各 有 侧重 ， 以 确保 大 
家 不 会 重复 提问 ， 并 全 面 考察 求职 者 的 能 力 水 平 。 
















































































面试 问题 主要 分 为 算法 、 编 程 水 平 、 软 件 架构 /设计 
能 力 等 几 大 块 ， 同时， 面试 官 也 会 考察 你 能 否 适应 
Facebook 快 节奏 的 开发 环境 。 

面试 过 后 ， 在 交流 你 的 表现 之 前 ， 面 过 你 的 面 
试 官 会 完 提交 书面 评价 报告 。 这么 做 是 为 了 确保 各 
位 面试 官能 对 你 的 表现 作出 相对 独立 的 评价 。 

一 旦 收 到 所 有 的 评价 报告 ,面试 小 组 和 招聘 经 
理 便 会 商讨 你 的 面试 结果 。 他 们 会 先 达 成 统一 意 
见 ， 然 后 提交 给 招聘 委员 会 。 

Facebook 很 看 重 “ 忍 术 ”( 灵活 应 变 ) 一 一 也 就 
是 使 用 任何 语言 快速 构建 优雅 、 可 扩展 解决 方案 的 
能 力 。 懂 PHP 并 不 会 显得 特别 突出 ， 因 为 Facebook 
也 有 很 多 后 台 工 作 要 用 到 C++、Python、Erlang 和 其 
他 语言 。 





















































Qa 感 兴趣 的 读者 可 以 访问 页 面 Facebook Engineering Puzzles: www.facebook.com/careers/puzzles.php。 一 一 译 者 注 
@ Facebook 总 部 位 于 美国 加 利 福 尼 亚 州 的 门 罗 帕 克 市 ， 地 址 为 黑客 路 1 号 ( 1 Hacker Way )。 
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必要 准备 事项 

作为 网 络 科技 的 新 贵 及 “当红 炸 子 
鸡 ”，Facebook 也 更 青睐 那些 富有 创业 
精神 的 开发 人 员 。 在 面试 过 程 中 ,你 要 
展现 出 自己 热衷 创造 新 事物 的 激情 。 


独特 之 处 

Facebook 由 公司 统一 招聘 员工 , 而 
不 是 专门 针对 某 个 团队 。 面试 成 功 并 入 
职 后 ， 你 会 先 参加 为 期 6 周 的 “新 兵 训 
练 营 ”, 帮 你 快速 适应 大 规模 的 代码 库 。 
资深 工程 师 会 担任 你 的 导师 , 辅导 你 党 
握 最 佳 实践 和 必 备 技能 ,最终 让 你 可 以 
游 丸 有 余地 加 入 自己 喜欢 的 项 目 组 。 





译 者 注 
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2.6 ”雅虎 面试 


雅虎 往往 只 招 美国 排名 前 20 的 高 校 毕业 生 , 不 过 其 他 求职 者 仍 可 通过 雅虎 公开 招聘 渠道 (或 
者 ， 可 以 内 部 推荐 的 话 就 更 好 了 ) 得 到 面试 机 会 。 取 得 面试 资格 后 ， 你 会 先 接受 一 轮 电话 面试 。 
对 你 进行 电话 面试 的 一 般 是 资深 员工 ， 比 如 技术 主管 或 经 理 。 

在 现场 面试 中 , 一 般 由 来 自 同一 团队 的 六 七 个 人 来 面试 你 , 每 轮 面试 时 长 45 分 钟 。 每 个 面试 
官 都 会 侧重 不 同 的 领域 。 比 如 ， 有 的 面试 官 可 能 侧重 于 数据 库 知 识 , 而 有 的 面试 官 则 会 关注 你 对 



































计算 机 体系 结构 的 理解 。 每 轮 面试 的 时 间 安 排 大 致 如 下 。 





口 开头 5 分 钟 : 


























口 最 后 20 分 钟 : 系统 设计 问题 。 比 如 , 设计 一 
个 大 型 分 布 式 缓存 系统 。 这 些 问 题 往往 与 你 
以 往 的 项 目 经 历 或 面试 官 当前 在 做 的 工作 
有 关 。 

当天 面试 结束 后 ， 你 可 能 还 会 跟 项 目 经 理 或 其 

他 人 面谈 一 次 。 内 容 包括 产品 展示 、 你 对 雅虎 的 疑 

虑 以 及 你 手 上 有 无 其 他 公司 的 录用 通知 ， 等 等 。 这 

次 面谈 旨 在 增进 双方 了 解 ,， 通 常 不 会 影响 你 的 面试 

结果 。 

与 此 同时 ,之 前 的 面试 官 会 讨论 你 的 表现 并 淮 

试 作 出 结论 。 最 终 录 用 与 否 由 招聘 经 理 决定 ， 他 会 

综合 考虑 面试 官 对 你 的 正面 及 负面 评价 。 

如 果 你 的 表现 不 错 ， 有 可 能 当天 就 会 收 到 口头 

录用 通知 。 但 也 不 一 定 。 也 许 他 们 要 过 几 天 才 通 知 

你 ,个 中 原因 不 一 ， 比 如 ， 你 应 聘 的 团队 可 能 还 想 

再 面试 几 个 人 看 看 。 









































役 对 话 。 比 如 ， 自 我 介绍 ， 聊 聊 项 目 经 历 等 。 
口 中 间 20 分 钟 : 编程 问题 。 比 如 ， 实 现 归并 排序 。 


必要 准备 事项 

雅虎 面试 少不了 系统 设计 问题 , 几 
乎 成 了 惯例 ， 所 以 ,还 请 做 好 相应 的 准 
备 。 他们 想 要 确认 你 不 仅 会 写 代码 , 而 
且 还 能 设计 软件 。 要 是 没有 这 方面 的 知 
识 , 也 不 要 紧 ， 你 仍然 可 以 给 出 自己 的 


设计 思路 。 


独特 之 处 

雅虎 的 电话 面试 一 般 由 拥有 决定 
权 的 人 负责 ， 比 如 招聘 经 理 。 此 外 ， 雅 
虎 往往 会 在 当天 给 出 面试 结果 ( 如 果 你 
能 入 他 们 法 眼 )， 这 一 点 很 特别 。 在 你 
进行 最 后 一 轮 面试 的 同时 , 其 他 面试 官 
也 正在 讨论 你 的 表现 。 
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“特殊 情况 


口 有 工作 经 验 的 求职 
口 测试 人 员 及 SDET 

口 项 目 经 理 与 产品 经 理 
口 技术 主管 与 部 门 经 理 
口 创业 公司 的 面试 


3.1 有 工作 经 验 的 求职 


只 要 你 仔细 读 过 之 前 的 章节 ,， 遇 到 以 下 情况 应 该 也 不 会 太 惊讶 : 在 面试 中 , 对 于 有 工作 经 验 
的 求职 者 和 初出 茅 庐 的 新 手 ， 面 试 官 会 问 同样 的 问题 ， 而 且 面 试 标准 差别 也 不 大 。 

你 可 能 知道 , 大 多 数 面试 题 都 是 些 涉 及 数据 结构 与 算法 的 常见 问题 。 多 数 公 司 认为 这 是 检验 
个 人 能 力 的 上 佳 手段 ， 故 而 对 所 有 求职 者 一 视 同仁 。 

有 些 面 试 官 可 能 会 对 有 工作 经 验 的 求职 者 稍稍 提高 标准 和 要 求 。 毕 竟 , 他 们 有 多 年 的 工作 经 
验 ， 理 应 比 新 手表 现 得 更 出 色 ， 不 是 吗 ? 

不 过 , 也 有 一 些 面试 官 持 相 反 的 看 法 。 有 工作 经 验 的 求职 者 离开 学 校 太 久 了 ,可 能 毕业 后 就 
没 怎么 接触 过 这 些 基 本 概念 。 他 们 忘记 其 中 一 些 细节 也 在 情理 之 中 , 所 以 我 们 应 该 稍微 降低 标准 。 

总 体 来 看 ， 两 者 相抵 。 所 以 ， 如 果 你 是 有 工作 经 验 的 求职 者 ， 碰 到 的 问题 和 面试 标准 基本 上 
与 新 手相 差 无 几 。 不 同 之 处 在 于 系统 设计 和 架构 方面 以 及 与 你 简历 相关 的 问题 。 

一 般 来 说 ,学 生 在 系统 架构 方面 没有 什么 积累 ， 这 类 经 验 只 有 通过 实践 才能 获得 。 因 此 , 面 
试 官 会 根据 你 的 经 验 水 平 来 评估 你 在 这 些 问 题 上 的 表现 。 当 然 , 在 校生 和 应 届 毕 业 生 也 会 被 问 及 
这 方面 的 问题 ， 总 之 都 要 竭尽 全 力 做 好 准备 。 

此 外 ， 对 于 “说 说 你 碰 到 过 的 最 棘手 的 bug? ”之 类 的 问题 ， 面 试 官 往往 期 竺 有 工作 经 验 者 
给 出 更 加 深入 、 证 人 印象 深刻 的 答案 。 你 拥有 更 丰富 的 经 验 ， 回 答 自 当 不 同 几 响 。 


3.2 测试 人 员 及 SDET 


软件 开发 测试 工程 师 (SDET ) 这 个 职位 确实 比较 复杂 。 作为 SDET, 不 仅 要 写 得 一 手 好 代码 ， 
是 优秀 的 测试 人 员 。 



























































险 
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建议 大 家 从 以 下 几 点 人 手 准备 SDET 的 面试 。 

口 准备 核心 测试 问题 : 例如 , 怎么 测试 一 只 灯泡 ?一 支 笔 ? 一 台 收 银 机 ? 抑或 是 微软 的 Word 

软件 ? 参看 本 书 “ 测 试 ”一 节 ， 有 助 于 你 在 这 些 问题 上 准备 得 更 充分 。 

D 练习 编程 问题 : 应 聘 SDET 被 拒 的 最 大 原因 就 是 编程 能 力 不 足 。 尽 管 这 个 职位 对 编程 能 
的 要 求 比 SDE ( 软件 开发 工程 师 ) 略 低 ， 但 面试 官 还 是 期 待 SDET 具 备 很 强 的 编程 能 力 和 
算法 功底 。 准 备 过 程 中 ， 不 妨 拿 针对 普通 开发 人 员 的 编程 和 算法 题 来 练 手 。 

D 练习 测试 编码 问题 对 SDET 来 说 ， 这 类 问题 的 常见 问 法 是 “ 写 代码 实现 X 功 能 "， 紧 接着 
就 是 ,“ 好 ， 请 测试 你 写 的 代码 "。 就 算 面试 官 没有 提 这 个 要 求 ， 你 也 应 该 问 问 自己 : “我 
该 如 何 测试 这 段 代 码 ? ”切记 : SDET 可 能 碰 到 任何 问题 ! 
对 测试 人 员 来 说 , 具备 良好 的 沟通 能 力也 非常 重要 , 因为 这 份 工作 要 求 你 跟 各 种 各 样 的 人 打 
交道 。 因 此 ， 不 要 对 行为 面试 题 掉以轻心 ， 可 参看 “行为 面试 题 ” 一 音 。 
职业 生涯 建议 
最 后 ， 提 几 点 职业 生涯 建议 ， 如 果 你 跟 许多 求职 者 一 样 ， 认 为 应 聘 SDET 职 位 是 进入 一 家 公 

司 的 “捷径 ”， 那 就 必须 想 清楚 ， 从 SDET 转 开发 岗位 可 不 轻松 。 假 如 你 有 此 意图 务必 加 强 自己 

的 编程 能 力 和 算法 功底 ， 并 尽 可 能 在 一 两 年 内 转岗 。 否 则 ,“ 温 水 者 青蛙 "， 拖 得 越久 ， 你 的 目标 

就 越 难以 实现 。 

总 之 ， 常 写 代码 ， 以 防 手 生 。 


















































3.3 项 目 经 理 与 产品 经 理 





不 同 公司 的 PM 职 位 大 相 径 庭 ， 甚 至 在 同一 家 公司 都 可 能 大 不 相同 。 例 如 ， 微 软 有 些 PM 职 位 
其 实 相 当 于 “口碑 传道 者 "， 职 责 是 面向 客户 推广 公司 产品 ， 有 点 接近 市 场 营销 。 然 而 ， 微 软 内 
部 的 其 他 PM 则 可 能 每 天 要 花 大 量 时 间 编 写 代 码 。 后 一 种 PM 在 面试 中 很 可 能 会 被 问 到 编码 问题 ， 
因为 这 是 其 工作 职责 的 重要 部 分 。 
大 体 上 ,求职 者 应 聘 PM 职 位 时 ， 面 试 官 主要 考察 以 下 儿 个 方面 。 
口 处 理 含糊 情况 : 虽然 它 不 是 面试 中 最 重要 的 考察 面 ,但 你 要 明白 面试 官 的 确 很 看 重 此 技 
能 。 他 们 想 看 到 你 面 对 含 糊 情况 不 会 手忙脚乱 、 不 知 所 措 ; 希望 看 到 你 迎 难 而 上 ， 比 如 
寻找 新 的 信息 、 优 先 考虑 最 重要 的 模块 ， 并 以 有 条 理 的 方式 解决 问题 。 面 试 官 一 般 不 会 
直接 考察 你 这 方面 的 能 力 〈 但 也 不 排除 这 种 可 能 性 )， 不 过 他 们 可 能 会 根据 你 在 处 理 问题 
时 的 表现 对 你 进行 评 佑 。 
口 以 客户 为 中 心 〈 态 度 层面 ) : 面试 官 希 望 看 到 你 能 做 到 以 客户 为 中 心 。 你 是 会 照搬 自己 
的 经 验 主观 腾 测 客户 使 用 产品 的 方式 ， 还 是 会 站 在 客户 的 立场 来 了 解 他 们 希望 如 何 使 用 
二 品 ? 诸如 “为 育 人 设计 一 款 闹钟 ”的 面试 题 考查 的 正 是 这 个 方面 。 当 你 听 到 这 类 面试 
题 时 ， 务 必 多 提问 题 以 了 解 产 品 主要 面向 哪些 客户 ， 以 及 他 们 会 如 何 使 用 该 产品 。 本 书 
“测试 ”一 节 有 很 多 相关 内 容 可 供 参 考 。 
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口 以 客户 为 中 心 〈 技 术 层 面 ) : 有 些 团队 做 的 产品 功能 非常 复杂 ， 要 求 PM 求 职 者 必须 充分 
掌握 相关 产品 ， 因 为 等 到 工作 时 再 上 手 是 来 不 及 的 。 欲 在 MSN Messenger 团 队 中 谋 得 PM 
一 职 ， 也 许 不 一 定 要 精通 即时 通讯 工具 ， 而 从 事 Windows Security 工 作 则 可 能 要 求 你 具备 
扎实 的 计算 机 安全 功底 。 因 此 ， 除 非 掌 握 了 必 备 技能 ， 否 则 在 面试 之 前 你 还 是 三 思 而 后 
行 吧 ! 

口 多 层次 交流 能 力 : PM 需 要 跟 公 司 内 各 个 级 别 、 跨 部 门 跨 职能 人 士 打交道 。 所 以 ， 面 试 官 

会 希望 你 具备 多 层次 交流 能 力 。 这 方面 的 考查 非常 直接 ， 比 如 ， 面试 官 会 抛 出 类 似 “向 

你 的 祖母 解释 什么 叫 TCP/ 耻 ”的 问题 。 当 然 ， 从 你 如 何 描述 此 前 的 项 目 经 历 ， 他 们 也 能 

看 出 你 的 沟通 能 力 。 

对 技术 的 热情 : 快乐 工作 的 员工 往往 是 高 产 员 工 ， 所 以 公司 要 确保 你 喜欢 并 享受 这 份 工 

作 。 在 你 的 回答 中 ， 应 该 处 处 展示 自己 对 技术 的 热情 ， 同 时 ， 要 是 能 对 公司 或 团队 充满 

热情 就 更 好 了 。 面 试 官 可 能 会 直接 问 你 :“ 为 什么 想来 微软 工作 ? ”此 外 ， 他 们 也 乐于 见 

到 你 充满 激情 地 描述 自己 此 前 的 工作 经 历 和 遇 到 过 的 挑战 。 面 试 官 喜欢 那些 不 惧 挑 战 并 

迎 难 而 上 的 求职 者 。 

团队 合作 /领导 能 力 : 这 大 概 是 PM 面试 中 最 重要 的 方面 , 无 颖 也 是 这 份 工作 本 身 的 关键 所 

在 。 所 有 面试 官 都 会 评估 你 能 和 否 与 其 他 人 合作 无 间 。 他 们 和 常会 提出 这 类 问题 :“ 说 说 你 怎 

么 处 理 团 队 成 员 没 能 按 进度 完成 工作 的 情况 。” 此 外 ,面试 官 也 想 了 解 你 能 否 妥 善 处 理 冲 

突 、 是 否 积极 主动 、 是 否 了 解 你 身边 的 人 ， 以 及 人 们 喜 不 喜欢 与 你 共事 。 你 在 “行为 问 

题 ” 上 所 做 的 准备 在 这 里 就 显得 尤为 重要 。 

以 上 这 些 方面 都 是 PM 的 必 备 技能 ， 因 此 也 是 面试 的 重点 。 各 个 方面 的 权重 大 致 取决 于 你 应 

聘 的 PM 职 位 以 及 该 职位 具体 看 重 哪些 方面 。 


3.4 技术 主管 与 部 门 经 理 


基本 上 , 技术 主管 职位 都 要 求 具 备 很 强 的 编程 技能 ， 部 门 经 理 职位 往往 也 不 例外 。 如 果 这 份 
工作 需要 编写 代码 ， 那 你 就 必须 具备 很 强 的 编码 技能 和 算法 功底 一 一 要 求 不 比 普通 开发 人 员 低 。 
特别 是 谷歌 ， 在 编程 上 ， 对 部 门 经 理 的 要 求 很 高 。 

此 外 ， 你 还 要 做 好 以 下 准备 。 

口 团队 合作 /领导 能 力 : 任何 担任 管理 类 角色 的 人 都 必须 懂得 团队 合作 ， 并 能 领导 员工 。 面 

试 官 会 或 明 或 瞳 地 考察 你 是 否 具备 这 些 能 力 。 一 方面 ， 他 们 会 直接 询问 你 在 此 前 工作 中 是 
如 何 处 理 冲 突 的 ， 比 如 你 与 主管 意见 相左 的 时 候 ; 另 一 方面 ， 面 试 官 也 会 暗中 观察 你 怎么 
与 他 们 互动 。 如 果 你 的 态度 过 于 傲慢 或 太 顺 从 ， 那 他 们 就 会 认为 你 不 太 适 合 当 管理 人 员 。 

口 把 握 轻重 缓急 : 管理 人 员 经 常 要 面 对 层 出 不 穷 的 状况 ， 比 如 怎样 才能 确保 团队 在 即将 到 

来 的 截止 期 前 完成 工作 。 你 需要 充分 展示 你 在 一 个 项 目 中 分 得 清 轻重 缓急 ， 砍 掉 无 足 轻 
重 的 部 分 。 把 握 轻 重 缓急 意味 着 要 通过 正确 的 提问 来 掌握 哪些 方面 至 关 重要 ， 以 及 合理 
预 佑 出 都 能 实现 哪些 方面 。 
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18 第 3 章 特殊 情况 








口 沟通 能 力 : 管理 人 员 不 仅 需 要 与 上 下 级 沟通 ， 而 且 可 能 还 会 与 客户 或 其 他 不 太 懂 技术 的 
人 进行 交流 。 面 试 官 希 望 看 到 你 具备 与 各 种 人 打交道 的 能 力 ， 跟 他 们 沟通 起 来 游 也 有 余 。 
实际 上 ， 面 试 官 也 是 在 拐弯 抹 角 地 评 佑 你 的 个 性 。 

口 “把 事情 做 好 ”的 能 力 : 经 理 与 主管 最 重要 的 职责 也 许 就 是 “把 事情 做 好 ”。 这 意味 着 你 
要 在 项 目 准 备 和 具体 实施 之 间 达 成 适当 的 平衡 。 你 需要 掌握 如 何 组 织 项 目 ， 以 及 如 何 激 
励 员 工 ， 从 而 达成 团队 目标 。 

最 终 ， 这 些 方面 大 都 会 跟 你 的 过 往 经 验 和 个 性 关联 起 来 。 务必 利用 “面试 准备 表格 ”做 好 充 

分 准备 。 


3.5 创业 公司 的 面试 


创业 公司 (start-up ) 的 职位 申请 和 面试 流程 千差万别 。 我 们 没 办 法 述 及 每 一 家 创业 公司 的 情 
况 ， 好 在 还 能 列举 一 些 共通 之 处 。 不 过 ， 也 请 理解 ， 实 际 情况 可 能 会 有 所 不 同 。 

1. 职位 申请 

很 多 创业 公司 都 会 在 网 上 发 布 招聘 启事 , 但 对 于 那些 最 热门 的 创业 公司 , 最 好 的 申请 方式 是 
通过 内 部 推荐 。 这 个 推荐 人 不 必 非 得 是 你 的 密友 或 同事 。 你 可 以 四 处 撒 网 ， 向 认识 的 人 表达 自己 
的 意向 ， 然 后 也 许 有 人 会 拿 起 你 的 简历 看 看 你 是 不 是 合适 人 选 。 

2. 签证 与 工作 许可 

很 遗憾 , 美国 大 多 数 小 型 创业 公司 没有 能 力 为 你 申请 工作 签证 。 他 们 跟 你 一 样 痛恨 劳工 部 教 
条 的 制度 ,可 还 是 无 能 为 力 。 如 果 你 没有 合法 身份 ， 同时 又 想到 创业 公司 工作 ,也 许 最 好 的 选择 
就 是 找 一 家 为 创业 公司 输送 人 才 的 专业 人 力 资源 代理 机 构 ， 又 或 者 , 你 可 以 盯 着 那些 规模 较 大 的 
初创 公司 。 

3. 简历 筛选 因素 

创业 公司 需要 的 工程 师 不 仅 要 聪明 过 人 ,会 写 代码 , 而 且 同 时 也 能 在 创业 环境 中 卖力 地 工作 。 
你 的 简历 应 该 展示 这 些 特质 。 

此 外 ， 你 还 必须 充满 干劲 ， 积 极 做 到 最 好 ; 这 些 创业 公司 急需 立马 能 上 手 干 活 的 员工 。 

4. 面试 流程 

与 大 公司 注重 你 在 软件 开发 上 的 整体 职业 素养 相 比 , 创业 公司 更 注重 你 的 个 性 契合 度 、 技 术 
技能 和 此 前 的 工作 经 验 。 
D 个 性 契合 度 : 面试 官 会 通过 你 与 他 们 的 互动 来 评估 你 的 个 性 契合 度 。 请 注意 ,与 面试 官 
交流 时 要 友善 、 专 注 ， 这 会 给 人 留 下 好 印象 ， 从 而 获得 更 多 工作 机 会 。 
口 技术 技能 : 创业 公司 需要 立马 能 上 手 干 活 的 人 ， 因 此 非常 看 重 你 在 特定 编程 语言 上 的 能 
力 。 如 果 你 恰好 掌握 该 公司 使 用 的 编程 语言 ， 请 务必 好 好 准备 与 此 相关 的 各 种 细节 问题 。 
口 以 往 经 验 : 创业 公司 会 问 你 很 多 以 往 工 作 经 验 有 关 的 问题 ， 请 特别 关注 “行为 面试 题 ” 

一 有 蛙 。o 
除 此 之 外 ， 你 还 会 碰 到 这 本 书 中 提 及 的 很 多 编程 及 算法 问题 。 
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口 积累 相关 经 验 
口 构建 人 际 网 络 
口 写 好 简历 








4.1 积累 相关 经 验 


录用 与 否 主 要 取决 于 你 在 面试 中 的 表现 , 而 简历 和 过 往 经 验 则 决定 你 有 没有 面试 机 会 。 你 应 
该 想方设法 提升 自己 的 技术 (及 非 技术 ) 水 平 。 不 管 是 应 届 毕 业 生 还 是 专业 人 士 , 拥有 额外 的 编 
程 经 验 都 会 让 你 受益 菲 浅 。 

在 校生 可 以 采取 下 面 这 些 举 措 。 

口 选修 有 大 作业 的 课程 : 如 果 你 还 是 学 生 ， 请 不 要 避 开 那些 有 大 作业 的 课程 。 将 来 ， 你 可 

以 把 这 些 项 目 经 历 都 写 在 简历 上 ， 这 会 大 幅 提 高 得 到 顶尖 科技 公司 面试 机 会 的 几率 。 当 
然 ， 这 些 项 目 与 实际 情况 联系 越 紧密 ， 效 果 就 越 好 。 

口 找 一 些 实习 生 工 作 : 就 算 你 是 大 学 新 生 ， 也 有 机 会 得 到 相关 的 专业 经 验 。 大 一 、 大 二 的 
学 生 可 以 考虑 参加 诸如 “微软 探索 者 ”和 “谷歌 编程 夏令 营 ” 这 样 的 活动 。 如 果 得 不 到 
类 似 的 机 会 ， 进 入 创业 公司 历练 一 下 也 不 错 。 

口 开拓 一 些 业务 或 项 目 : 绝 大 多 数 公司 都 青睐 富有 创业 精神 的 人 。 此 举 不 仅 可 以 培养 一 些 
技术 经 验 ， 而 且 同 时 也 能 展示 你 的 主观 能 动 性 和 把 事情 做 好 的 能 力 。 你 可 以 利用 周末 和 
休息 时 间 写 个 软件 。 要 是 认识 学 校 教 授 ， 不 妨 试 着 请 他 予以 “资助 "， 以 便 你 将 自己 的 工 
作 变 成 一 项 独立 研究 。 

另 一 方面 ， 专 业 人 士 可 能 早已 累积 好 相应 资本 ， 准 备 跳槽 进入 他 们 梦 朵 以 求 的 公司 。 比 如 ， 
谷歌 的 开发 人 员 可 能 已 经 攒 够 经 验 ， 有 机 会 跳槽 到 Facebook 工 作 。 不 过 ， 如 果 你 想 从 不 知名 的 小 
公司 跳 到 科技 巨头 公司 ， 或 者 从 测试 岗位 转 成 开发 人 员 ， 请 参考 以 下 这 些 建议 。 

口 多 承担 一 些 编程 职责 : 在 不 透露 跳槽 意向 的 前 提 下 ， 你 可 以 向 经 理 表 达 自 己 想 在 编程 上 

接受 更 大 的 挑战 。 尽 可 能 地 参与 一 些 重 大 项 目 ， 并 多 多 使 用 对 自己 以 后 有 利 的 技术 ,将 
来 它们 会 成 为 简历 上 的 亮点 。 男 外 ,简历 上 也 要 尽量 多 列举 这 些 与 编程 相关 的 项 目 。 
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20 第 4 章 面试 之 前 








口 善 用 晚上 和 周末 的 闲暇 时 光 : 如 有 空闲 时 间 ， 可 以 试 着 构建 一 些 手 机 应 用 、 网 页 应 用 或 
桌面 软件 。 这 样 ， 你 就 有 机 会 接触 到 时 下 流行 的 新 技术 ， 从 而 更 契合 科技 公司 的 要 求 。 
这 些 项 目 经 验 都 可 以 写 到 简历 上 ， 没 有 什么 比 “ 为 兴趣 而 工作 ”更 能 打动 招聘 人 员 的 了 。 
总 而 言 之 , 公司 最 青睐 的 人 才 必 须 具备 两 大 特性 : 一 是 天 资 聪 突 ， 二 是 扎实 的 编程 功底 。 要 
是 你 能 在 简历 上 充分 展示 这 两 点 ， 面 试 机 会 就 唾 手 可 得 了 。 
此 外 ,你 应 当 提 前 规划 好 职业 发 展 路 径 。 如 果 打 算 转 型 成 为 管理 者 ,哪怕 当下 应 聘 的 仍 是 开 
发 轴 位 ， 也 应 现在 就 想方设法 培养 自己 的 领导 才能 。 


4.2 构建 人 际 网 络 


你 或 许 听 说 过 很 多 人 靠 朋友 推荐 找到 了 好 工作 。 不 过 , 你 可 能 想不到 , 还 有 更 多 人 是 通过 朋 
友 的 朋友 找到 工作 的 。 这 真 的 很 有 道理 。 用 极 客 的 话 来 说 ， 你 有 N 个 朋友 ， 也 就 意味 着 你 有 N 个 
朋友 的 朋友 。 

那么 , 在 你 找 工作 时 这 个 数字 意味 着 什么 呢 ? 这 意味 着 , 不 管 是 直接 联系 人 还 是 拐弯 抹 角 的 

关系 ， 对 你 找 工 作 都 很 有 帮助 。 

1. 什么 叫好 的 人 际 网 络 

好 的 人 际 网 络 不 仅 意味 着 你 广 交 朋 友 (广度 ) 还 要 与 他 们 保持 紧密 的 联系 (深度 ) 这 名 话 

看 似 矛 盾 ， 实 则 要 辩证 地 看 待 。 
口 广度 : 你 的 人 际 网 络 中 不 仅 要 有 业内 技术 人 士 ， 而 且 最 好 还 能 涵盖 各 行 各 业 的 人 才 。 比 
如 说 ， 结 交 一 位 会 计 朋 友 会 对 你 的 职业 生涯 帮助 很 大 ， 因 为 他 很 可 能 在 其 他 领域 有 很 多 
朋友 。 有 时 候 ， 其 中 有 些 人 可 能 就 想 认识 像 你 这 样 的 技术 人 才 。 请 抱 着 开放 的 交友 态度 
去 对 竺 他人。 
口 深度 : 通过 自己 的 密友 来 结交 新 朋友 是 个 不 错 的 方法 ， 总 好 过 让 不 太 熟 的 人 为 你 牵线 搭 
桥 。 此 外 ， 人 们 会 对 那些 所 谓 的 “ 老 油 条 ”和 “交际 花 ” 避 之 唯恐 不 及 ， 觉 得 这 些 人 太 
虚伪 了 。 因 此 ， 尽 量 与 朋友 保持 真诚 和 深厚 的 关系 。 
其 中 的 微妙 之 处 就 在 于 找到 平衡 点 ,你 认识 的 人 当然 越 多 越 好 , 但 要 确保 自己 待人 真诚 、 开 
放 。 如 果 只 是 热衷 于 收集 大 家 的 名 片 ， 那 你 最 终 往往 只 会 一 无 所 获 。 

2. 如 何 构建 坚实 的 人 际 网 络 

有 些 人 认为 ,我 们 应 当 走 出 家 门 ， 去 结识 更 多 人 。 这 么 说 也 有 道理 。 但 是 去 哪里 呢 ? 而 且 ， 
如 何 才能 将 “点 头 之 交 ” 发 展 成 好 朋友 呢 ? 

以 下 这 些 建议 或 许 能 给 你 一 些 启 发 。 

(1) 通过 Meetup.com 这 样 的 社交 网 站 或 校友 网 来 获取 你 感 兴趣 的 活动 资讯 。 记 得 带 上 你 的 名 
片 。 如 果 你 暂时 没有 工作 或 还 是 学 生 ， 那 就 自己 印 些 名 片 。 

(2) 主动 跟 人 打招呼 。 也 许 你 生性 胆 愤 ， 不 敢 迈 出 这 第 一 步 。 但 请 相信 我 ， 没 人 会 拒绝 你 的 
友好 之 举 ， 其 至 有 些 人 还 会 欣赏 你 的 自信 。 话 说 回来 ， 最 坏 能 坏 到 什么 地 步 呢 ? 他 们 不 喜欢 你 ， 
不 会 与 你 结交 ， 从 此 和 你 老死 不 相 往 来 吗 ? 
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(3) 大 大 方 方 地 聊 你 的 兴趣 ， 并 和 人 们 谈论 他 们 的 兴趣 。 如 果 他 们 正在 运营 创业 公司 ， 或 是 
从 事 其 他 你 也 感 兴趣 的 活动 ,不 妨 邀 请 他 们 一 起 喝 咖啡 继续 畅谈 。 

(4) 活动 结束 后 ,你 可 以 在 LinkedIm 上 把 他 们 加 为 好 友 , 或 者 给 他 们 发 邮件 。 当 然 , 更 好 的 方 
式 就 是 邀请 他 们 一 起 喝 咖啡 , 这 样 你 们 就 会 有 充足 的 时 间 来 畅谈 他 们 的 创业 公司 , 或 是 双方 都 感 
兴趣 的 话题 。 

(5) 最 重要 的 是 乐于 助人 。 经 常 助人 一 臂 之 力 ， 你 就 会 给 人 留 下 慷慨 大 方 、 友 好 和 善 的 印象 。 
那些 乐善好施 的 人 往往 也 会 得 到 更 多 帮助 。 

切记 , 不 要 只 局 限于 现实 生活 中 的 社交 。 在 这 个 信息 爆炸 的 时 代 , 社交 还 可 以 拓展 到 网 络 上 ， 
通过 博客 、 微 博 、Facebook 和 电子 邮件 结交 朋友 。 

当然 ， 也 不 要 太 “ 走 火 入 魔 ” 沉 迷 于 在 线 社交 ， 你 得 努力 建立 实 实在 在 的 人 际 关系 。 


4.3” 写 好 简历 


简历 筛选 标准 与 面试 标准 并 无 太 大 差别 ， 也 是 看 你 是 否 又 聪明 又 会 写 程序 。 

这 意味 着 你 在 准备 简历 时 应 该 突出 这 两 点 。 提 到 自己 喜欢 打 网 球 、 旅 游 或 玩 魔法 牌 可 没 
什么 用 。 在 罗列 这 类 无 关 紧 要 的 爱好 之 前 ， 务 请 三 思 ， 宝 贵 的 篇 幅 应 该 用 来 展示 自己 的 技术 
才能 。 

1. 简历 篇 幅 长 度 适 中 

在 美国 ， 人 们 会 建议 工作 经 验 不 足 10 年 的 求职 者 将 简历 压缩 成 一 页 ; 超过 10 年 的 ,至 多 用 两 
页 。 为 什么 呢 ? 主要 有 两 大 理由 。 

口 招聘 人 员 浏 览 一 份 简历 一 般 只 会 用 20 秒 钟 左 右 。 要 是 你 的 简历 言 简 意 凡 恰 到 好 处 ， 招 聘 
人 员 一 眼 就 能 看 到 。 废 话 连篇 只 会 模糊 重点 ， 扰 乱 招 聘 人 员 的 注意 力 。 
口 有 些 人 遇 上 元 长 的 简历 连 看 都 不 看 。 你 真 的 想 冒 这 个 风险 , 让 别人 直接 扔 掉 你 的 简历 吗 ? 

如 果 看 到 这 里 你 还 在 想 , 我 工作 经 验 太 丰富 了 , 一 页 篇 幅 根 本 放 不 下 怎么 办 ? 相信 我 ， 你 可 
以 的 。 一 开始 大 家 都 会 这 人 么 说 。 其 实 ， 简 历 写 得 洋洋 洒洒 并 不 代表 你 经 验 丰富 ， 反 而 只 会 显得 你 
完全 抓 不 住 重点 。 

2. 工作 经 历 

简历 不 是 也 不 应 该 是 关于 工作 经 历 的 编 年 史 。 比 如 , 卖 过 冰淇淋 跟 聪 明 与 否 或 代码 写 得 怎么 
样 关系 不 大 。 你 应 该 只 列举 那些 相关 的 工作 经 验 。 















































































































































@ 列举 要 点 
在 描述 工作 经 历时 ， 请 尽量 采用 这 样 的 格式 :“ 使 用 Y 实 现 了 X， 从 而 达到 了 Z 效 果 。” 比如 ， 
下 面 这 个 例子 : 
口 “ 通 过 实施 分 布 式 缓存 功能 减少 了 75% 的 对 象 泻 当 时间， 从 而 使 得 用 户 登 录 速 度 加 快 
了 10%。” 


下 面 还 有 一 个 例子 ， 描 述 略 有 不 同 : 
口 “ 实 现 了 一 种 新 的 基于 windiff 的 比较 算法 ， 系 统 平均 匹配 精度 由 1.2 提 升 至 1.5。” 
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尽管 不 是 所 有 经 历 都 能 套用 此 名 型， 但 原则 无 二 : 描述 做 过 的 事情 、 怎 么 做 的 ， 以 及 结果 如 
何 。 理 想 的 做 法 是 尽 可 能 地 量化 结 

3. 项 目 经 历 

在 简历 中 列 出 “项 目 经 历 ” 这 一 部 分 会 让 你 看 起 来 很 专业 。 对 于 大 学 生 和 毕业 不 久 的 新 人 尤 
其 如 此 。 

简历 上 应 该 只 列举 2 到 4 个 最 重要 的 项 目 。 描 述 项 目 要 简明 扼要 ， 比 如 使 用 哪些 语言 和 技术 。 
你 也 可 以 加 上 一 些 细节 ， 比 如 该 项 目 是 个 人 独立 开发 还 是 团队 合作 的 成 果 , 是 某 一 门 课程 的 一 部 
分 还 是 自主 开发 的 。 当 然 ， 这 些 细节 不 一 定 放 到 简历 上 ， 除 非 能 让 简历 更 出 彩 。 

项 目 也 不 要 列 太 多 。 很 多 求职 者 就 犯 过 这 样 的 错误 ， 在 简历 上 一 股 脑 儿 列 出 先前 做 过 的 13 
个 项 目 ， 鱼 龙 混 条， 效果 反而 不 佳 。 

4. 编程 语言 和 软件 

@ 软件 

一 般 说 来 ,“ 熟 悉 微 软 Office” 之 类 不 必 列 人 简历 。 这 应 该 是 地 球 人 的 必 备 技能 ， 列 出 来 反而 
会 模糊 重点 。 你 应 该 列 出 那些 能 反映 自身 技术 水 平 的 软件 或 系统 ( 比如 Visual Studio 、Linux 等 )， 
不 过 坦白 说 ， 这 人 么 做 用 处 也 不 大 。 


@ 编程 语言 



































列举 编程 语言 确实 是 件 难 事 。 我们 到 底 应 该 列 出 自己 用 过 的 所 有 语言 , 还 是 只 列 那些 用 得 最 
顺手 的 语言 呢 ? 我 建议 采用 下 面 这 个 折 中 办 法 : 列 出 你 用 过 的 主要 语言 ， 后 面 加 上 熟练 程度 。 比 
如 像 下 面 这 样 : 





口 编程 语言 : Java ( 非常 熟练 )，C++ ( 熟练 )，JavaScript ( 有 过 使 用 经 验 )。 

5. 给 母语 为 非 英语 的 人 及 国际 人 士 的 建议 

一 些 公司 可 能 会 因为 小 小 的 笔 误 就 扔 掉 你 的 简历 , 所 以 请 至 少 找 一 位 以 英语 为 母语 的 人 来 帮 
你 审阅 简历 。 

此 外 ,申请 美国 的 工作 时 ， 简 历 中 不 要 包含 年 龄 、 婚 姻 状况 或 国籍 等 。 公 司 并 不 想 看 到 这 些 
个 人 信息 ， 因 为 习 巷 上 不 必要 的 麻烦 。 
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行为 面试 题 


口 准备 工作 
口 如 何 应 对 


5.1. 准备 工作 


行为 面试 题 的 考察 有 各 种 各 样 的 原因 。 人 们 可 以 通过 这 
地 掌握 你 的 履历 ， 又 或 者 缓和 一 下 面试 的 紧张 气氛 。 不 管 怎 
好 准备 、 有 的 放 矢 。 


行为 面试 题 一 般 是 这 么 问 的 :“ 说 说 你 曾经 ……” 面 试 官 可 能 还 会 要 求 你 列举 并 说 明 具 体 的 
项 目 或 岗位 。 我 建议 你 先 按 如 下 格式 拟定 一 份 “准备 表格 ”: 

常见 问题 项 目 1 项 目 2 项 目 3 项 目 4 
最 难 的 部 分 
有 什么 收获 
最 有 意思 的 部 分 
最 难 解 的 bug 
最 享受 的 过 程 


与 团队 成 员 的 冲突 


第 一 行 可 以 列举 你 在 简历 中 提 到 的 主要 事项 ， 比 如 项 目 、 职 位 或 活动 。 第 一 列 应 该 写 一 些 常 
ee 你 最 享受 和 最 不 喜欢 的 过 程 、 最 难 的 部 分 、 从 中 学 到 的 经 验 、 最 难 解 的 bug， 等 等 。 然 
， 在 对 应 单元 格 里 写 下 相应 的 小 故事 。 
当面 试 官 问 及 项 目 有 关 的 问题 时 ,你 就 能 回想 起 这 些小 故事 ,从 容 应 对 。 记 得 在 面试 前 复习 
这 份 表格 。 
另外， 建议 大 家 将 小 故事 浓缩 成 几 个 关键 字 ， 以 便 填 到 单元 格 里 。 这 样 一 来 ,这 份 表格 用 起 
来 就 会 更 顺手 ,方便 记忆 。 





些 问 题 来 了 解 你 的 个 性 , 或 是 更 深入 
样 ， 这 个 部 分 很 重要 ， 而 且 有 办 法 做 
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电话 面试 时 , 最 好 将 这 份 表格 摆 在 自己 跟前 。 把 每 个 小 故事 都 概括 成 几 个 关键 字 ， 更 容易 记 
忆 ， 自 然而 然 就 能 把 整个 故事 串 起 来 ， 比 死记 人 硬 背 一 段 文 字 要 轻松 得 多 。 

你 还 可 以 将 这 份 表格 扩展 成 一 系列 “ 软 问题 ”， 比 如 团队 冲突 、 项 目 失 败 的 经 历 以 及 你 需要 
说 服 团队 成 员 的 事例 。 对 于 那些 不 是 专职 开发 的 职位 如 技术 主管 、PM 或 测试 人 员 而 言 ， 这 些 都 
是 很 常见 的 面试 问题 。 如 果 你 刚好 要 申请 其 中 一 个 职位 ， 建 议 你 针对 这 些 “ 软 问题 ”再 准备 一 份 
表格 。 

在 回答 问题 时 ， 你 不 只 是 在 讲述 一 个 与 该 问题 密切 相关 的 故事 ， 更 是 在 向 别人 展现 自我 。 所 
以 ， 请 用 心思 索 每 个 故事 都 能 体现 出 自己 的 哪些 特性 。 

1. 你 有 哪些 缺点 

被 问 及 自己 有 哪些 缺点 时 ， 回 答 不 要 太空 泛 ! 诸如 “我 最 大 的 缺点 就 是 工作 太 努 力 了 ”的 回 
答 ， 反 而 会 显得 你 傲慢 自 大 ， 并且 不 愿 正视 自己 的 不 足 。 没有 人 喜欢 与 这 样 的 人 共事 。 因 此 ,你 
应 该 提 到 真实 、 合 乎 情理 的 缺点 ， 然 后 话 锋 一 转 ， 强 调 自己 如 何 克 服 这 些 缺 点 。 比 如 :“ 有 时 候 ， 
我 可 能 对 细节 不 够 重视 。 好 的 一 面 是 我 反应 迅速 、 执 行 力 强 ,， 但 不 免 会 粗心 大 意 而 犯错 。 有 鉴于 
此 ， 我 总 是 会 找 其 他 同事 帮忙 检查 自己 的 工作 ， 确 保 不 出 问题 。” 

2. 项 目 中 最 难处 理 的 问题 是 什么 

当面 试 官 问 到 这 个 问题 时 ， 请 不 要 泛泛 地 回答 “我 得 学 习 很 多 新 的 编程 语言 和 技术 ”。 除 非 
你 实在 是 无 话 可 说 ， 否 则 这 种 回答 似乎 是 在 强调 : 该 项 目 并 不 是 很 难 ， 没 什么 坏 手 的 问题 。 

3. 你 应 该 问 面试 官 哪些 问题 

大 多 数 面试 官 都 会 给 你 提问 的 机 会 。 有 意 无 意 间 , 你 提问 的 质量 也 会 成 为 他 们 评估 你 的 整体 
表现 的 因素 之 一 。 

也 许 你 会 在 面试 过 程 中 临时 想到 若干 问题 , 但 你 还 是 可 以 并 且 应 该 事先 准备 好 问题 。 对 公司 
和 团队 做 些 调研 ， 有 助 于 你 准备 问题 。 

问题 可 以 分 成 以 下 三 大 类 。 

@ 真实 的 问题 

也 就 是 你 真 的 想 知道 答案 的 问题 。 下 面 是 对 多 数 求职 者 有 用 的 一 些 问题 点 。 

(1)“ 你 每 天 有 和 多少 时间 花 在 写 代码 上 ?” 

(2)“ 你 一 周 要 开 几 次 会 ? ” 

(3)“ 整 个 团队 中 , 测试 人 员 、 开 发 人 员 和 项 目 经 理 的 比例 是 多 少 ? 他们 是 如 何 互 动 的 ?团队 
怎么 做 项 目 规 划 ? ” 

这 些 问 题 有 助 于 你 较 好 地 了 解 公司 的 工作 环境 和 日 程 安排 。 

@ 有 见地 的 问题 

有 见地 的 问题 可 以 充分 反映 出 你 的 编程 水 平和 技术 功底 , 同时, 还 能 显示 你 对 该 公司 或 其 产 
品 的 兴 

()“ 我 注意 到 你 们 使 用 了 X 技 术 ， 请 问 你 们 是 如 何 处 理 Y 问 题 的 ? ” 

(0)“ 为 什么 你 们 的 产品 选择 使 用 X 协 议 而 不 是 Y 协 议 ?” 据 我 所 知 ， 虽 然 X 有 A、B 、C 等 几 大 
好 处 ， 但 因为 存在 D 问 题 ， 很 多 公司 并 未 采用 该 协议 。 
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只 有 事先 对 该 公司 做 过 充分 调研 ， 才 问 得 出 这 类 有 深度 的 问题 。 

@ 富有 激情 的 问题 

这 些 问 题 旨 在 展示 你 对 技术 的 热忱 。 要 让 面试 官 知道 你 热衷 学 习 , 将 来 能 为 公司 的 发 展 做 出 
很 大 贡献 。 比 如 : 

(1)“ 我 对 可 扩展 性 很 感 兴趣 。 请 问 你 从 事 过 分 布 式 系统 方面 的 工作 吗 ? 有 哪些 机 会 可 以 学 习 
这 方面 的 知识 ?” 

(2)“ 我 对 X 技 术 不 是 太 熟 悉 ， 不 过 听 上 去 是 个 不 错 的 解决 方案 。 你 能 给 我 多 讲 讲 它 的 工作 原 
理 吗 ? ” 


5.2 如何 应 对 











如 前 所 述 ， 面 试 官 喜欢 在 面试 开始 和 结束 时 与 你 谈天 说 地 或 聊 歼 “ 软 技 能 "。 他 们 通常 
你 的 简历 问 些 问题 ， 或 者 泛泛 地 提问 ， 此 时 你 也 可 以 问 一 些 和 公司 有 关 的 问题 。 这 个 面试 环 
了 缓和 和 气氛， 也 是 意 在 了 解 你 。 

回答 这 类 问题 时 ， 切 记 以 下 几 个 建议 。 

1. 力求 具体 ， 切 忌 自 大 

骄傲 自 大 是 面试 大 尽 。 可 是 , 你 又 想 给 面试 官 留 下 深刻 的 印象 。 那么 ,怎样 才能 很 好 地 秀 出 
自己 的 实力 而 又 不 显 自 大 呢 ? 那 就 是 回答 问题 要 具体 ! 

具体 也 就 是 只 陈述 事实 ， 余 下 的 留 给 面试 官 自己 解读 。 请 看 下 面 这 个 例子 。 
口 一 号 求职 者 :“ 我 几乎 包揽 了 团队 中 所 有 累 活 和 难 活 。 
口 二 号 求职 者 :“ 我 实施 了 文件 系统 ， 因 为 XXXX 等 原因 ， 这 是 整个 项 目 中 最 难 的 一 部 分 。” 

二 号 求职 者 的 回答 不 仅 听 起 来 更 令 人 印象 深刻 ， 而 且 也 不 会 显得 骄傲 自 大 。 

2. 省 略 细 枝 末节 

当 求职 者 就 某 个 问题 唆 唆 不 体 时 ， 不 熟悉 该 主题 或 项 目的 面试 官 往 往 听 得 一 头 雾 水 。 所 以 ， 
请 省 略 细 枝 未 节 ， 只 谈 重 点 。 换 言 之 ， 建 议 你 这 么 回答 :“ 在 研究 最 常见 的 用 户 行为 并 应 用 
Rabin-Karp 算 法 "后 ,我 设计 了 一 种 新 算法 ， 在 90% 的 情况 下 搜索 操作 的 时 间 复 杂 度 由 O(n) 降 至 
O(log n)。 您 要 是 感 兴 趣 的 话 ， 我 可 以 详细 说 明 。” 该 回答 言 简 意 凡 ， 重点 突出 ; 要 是 面试 官 对 实 
现 细 节 感 兴趣 ， 他 会 主动 询问 。 

3. 回答 条 理 清 晰 

回答 行为 面试 题 有 两 种 常见 的 组 织 方式 : 主题 先行 法 与 S.A.R. 法 ">。 你 可 以 分 别 或 组 合 使 用 
这 两 种 技巧 。 

@ 主题 先行 

主题 先行 即 开 门 见 山 、 直 奔 主题 ， 回 答 简 洁 明 了 。 比 如 ， 











会 就 
节 除 


















































@ Rabin-Karp 算 法 是 由 Michael O. Rabin 和 Richard M. Karp 于 1987 年 提出 的 字符 串 匹配 算法 。 一 一 译 者 注 
@@) S.A.R 即 Situation 、Action 与 Result 的 缩写 ， 情 景 、 行 为 与 结果 ，。 编者 注 
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口 面试 官 :“ 讲 一 讲 你 必须 说 服 一 群 人 作出 大 幅 调 整 的 事例 。” 

口 求职 者 :“ 好 的 ， 我 在 学 校 提出 过 一 个 让 本 科 生 互相 授课 的 想法 ， 并 成 功 说 服 学 校 采纳 该 
建议 。 起 初 我 们 学 校规 定 ……” 

主题 先行 法 可 以 快速 抓 住 面试 官 的 注意 力 , 让 他 了 解 事情 梗概 。 此 外 ,假如 你 有 滔滔 不 绝 的 














倾向 ,这 也 有 助 于 你 不 偏离 主题 ， 因 为 你 早已 开门 见 山 地 点 明 主 由 。 


@ SA.R. 
S.A.R. 法 是 指 先 描述 情景 ,然后 解释 你 采取 的 行动 ， 最 后 陈述 结果 。 


示例 :“ 说 说 你 与 某 位 “ 刺 头 ”队友 相处 的 事例 。” 


口 情景 : 在 操作 系统 课 的 大 作业 中 ,我 被 安排 与 其 他 三 个 人 合作 。 其 中 两 人 都 很 卖力 ,但 
另外 一 个 人 做 的 不 多 。 开 会 时 他 总 是 沉默 寡言 ， 也 极 少 参与 邮件 讨论 ， 只 是 很 吃力 地 完 
成 分 配给 他 的 模块 。 

口 行动 : 有 一 天 课 后 ， 我 把 他 拉 到 一 边 讨论 这 门 课程 ， 然 后 谈 起 我 们 的 大 作业 。 我 坦诚 地 
询问 他 对 大 作业 的 感受 ， 以 及 他 最 感 兴趣 的 模块 。 他 建议 让 他 处 理 最 简单 的 几 个 模块 ， 
并 承诺 会 完成 最 后 的 总 结 报告 。 我 意识 到 他 其 实 一 点 都 不 懒 一 一 他 只 是 对 这 项 大 作业 感 
到 很 困惑 ， 并 且 缺 少 自信 心 。 此 后 ， 我 开始 与 他 合作 ， 进 一 步 细 分 组 件 模 块 。 此 外 ,在 
工作 中 我 还 经 常 称赞 他 以 增强 他 的 自信 心 。 

口 结果 : 他 依然 是 我 们 团队 最 弱 的 一 员 ， 但 是 进步 很 大 。 他 及 时 完成 了 分 配给 自己 的 任务 ， 
参与 讨论 也 更 积极 。 后 来 在 男 一 个 大 作业 中 ， 我 们 合作 得 非常 愉快 。 

切记 , 描述 情景 与 结果 务必 言 简 意 凡 。 面试 官 一 般 不 需要 太 多 细节 就 知道 来 龙 去 脉 , 实际 上 ， 

















细节 过 多 反而 会 令 他 们 摸 不 着 头脑 。 

















采用 S.A.R. 法 简明 扼要 地 描述 情景 行动 和 结果 , 可 让 面试 官 快速 了 解 你 是 如 何 施 加 影响 的 ， 





起 到 了 什么 作用 。 
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口 技术 准备 
口 如 何 应 对 

口 算法 题 的 五 种 解法 

口 怎样 才 算 好 代码 

6.1 技术 准备 

既然 你 买 了 这 本 书 ， 说 明 你 已 经 为 技术 面试 做 了 不 少 准备 。 王 得 好 ! 

即便 如 此 ， 准 备 方式 也 有 好 有 坏 。 许 多 求职 者 只 是 通读 一 遍 问题 和 解法 ， 罗 回春 京 。 这 好 比 
试图 单 赁 看 问题 和 解法 就 想 学 会 微 积 分 。 你 得 动手 练习 如 何 解 题 。 单 靠 死 记 硬 背 效果 不 彰 。 

1. 如 何 练习 

就 本 书 的 面试 题 ( 以 及 你 可 能 遇 到 的 其 他 题目 )， 请 参照 以 下 几 个 步骤 。 

(1) 尽量 独立 解 题 。 也 就 是 说 ， 要 试 着 实战 演练 解 题 过 程 。 许 多 题目 确实 很 难 , 但 是 没关系 ， 
不 要 怕 ! 此 外 ， 解 题 时 还 要 考虑 空间 和 时 间 效 率 。 多 问 问 自己 ， 能 否 通过 降低 空间 效率 来 提高 
时 间 效 率 ， 或 者 相反 。 

(2) 在 纸 上 编 写 算法 代码 。 之 前 你 一 直 在 计算 机 上 编写 代码 ， 习 惯 了 由 此 带 来 的 诸多 便利 。 
不 过 , 在 面试 中 ,你 可 享受 不 到 语法 高 亮 、 代 码 补 全 或 编译 构建 的 种 种 好 处 。 不 妨 在 纸 上 编 写 代 
码 模拟 面试 时 的 情景 。 

(3) 在 纸 上 测 试 代码 。 也 就 是 要 在 纸 上 写 下 一 般 用 例 、 基 本 用 例 和 错误 用 例 等 。 面 试 中 就 得 
这 么 做 ， 因 此 最 好 提前 做 好 准备 。 

(4) 将 代码 照 原样 输入 计算 机 。 你 也 许 会 犯 一 大 堆 错误 。 请 整理 一 份 清单 ， 罗 列 自己 犯 过 的 
所 有 错误 ， 这 样 在 真正 的 面试 中 才能 牢记 在 心 。 

此 外 ， 模 拟 面 试 (mock interview ) 也 非常 有 用 。CareerCup.com 提 供 了 与 微软 、 谷 歌 和 亚 马 
逊 等 公司 员工 进行 模拟 面试 的 机 会 ， 当 然 , 你 也 可 以 跟 朋友 一 起 演练 ,轮流 当面 试 官 给 对 方 做 模 
拟 面 试 。 你 的 朋友 不 见得 受过 什么 专业 训练 ， 但 至 少 还 能 带 你 过 一 遍 编 码 或 算法 面试 题 。 

2. 你 需要 掌握 的 知识 

大 多 数 面 试 官 都 不 会 问 你 二 又 树 平衡 的 具体 算法 或 其 他 复杂 算法 。 老 实说 , 离开 学 校 这 么 多 
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年 ， 恐 怕 他 们 自己 也 记 不 清 这 些 算法 了 。 
一 般 来 说 ， 你 只 要 掌握 基本 知识 即 可 。 下 面 这 份 清单 列 出 了 必须 掌握 的 知识 : 








数据 结构 
链表 朗 休 位 操作 
二 又 树 深度 人 单 例 设计 模式 
单词 查找 树 ( trie ) 查找 工厂 设计 模式 
栈 内 存 ( 栈 和 堆 ) 
队列 排 递归 
向 量 / 数 组 列表 重 人 /查找 等 大 0 时 间 
散 列 表 






























































对 于 上 述 各 项 主题 , 务必 掌握 它们 的 具体 实现 和 用 法 、 应 用 场景 、 空 间 和 时 间 复 杂 度 如 何等 。 

对 于 其 中 的 数据 结构 和 算法 ,你 还 要 练习 如 何 从 无 到 有 地 实现 。 面 试 官 可 能 会 要 求 你 直接 实 
现 一 种 数据 结构 或 算法 ， 或 者 对 其 进行 修改 。 不 管 怎样 ， 你 越 是 熟悉 具 体 实现 ， 把 握 就 越 大 。 

其 中 ， 散 列表 一 项 特别 重要 。 你 会 发 现 ， 解 决 面试 问题 时 ， 经 常会 用 到 散 列 表 。 

3. 2 的 震 表 

有 些 人 已 经 把 下 面 这 张 表 背 得 滚 瓜 烂 熟 ， 如果 你 还 没有 的 话 ， 面 试 前 一 定 要 背 下 来 。 回 答 可 
扩展 性 问题 时 ， 这 张 表 用 处 很 大 ， 借 助 它 可 以 快速 算出 一 组 数据 占用 多 少 空间 。 





















































准确 值 (X) 近似 值 X 字 节 转 换 成 MB、GB 等 
128 

256 

1024 

65 536 

1 048 576 
































1 073 741 824 
4294 967 296 
1 099 511 627 776 一 万 亿 (trillion ) 




















有 了 这 张 表 , 就 可 以 做 速算 。 例 如, 一 个 将 每 个 32 位 整数 映射 为 布尔 值 的 散 列表 可 以 把 一 台 
计算 机 的 内 存 填 满 。 

在 接受 互联 网 公司 的 电话 面试 时 ,不妨 将 这 张 表 放 在 跟前 ， 也 许 能 派 上 用 场 。 

4. 需要 知道 C++、Java 或 其 他 编程 语言 的 细节 了 吗 ? 

我 个 人 不 会 问 这 类 问题 ( 比如 “什么 是 虚 函 数 表 ”)， 不 过 许多 面试 官 确实 会 问 。 

对 于 微软 、 谷 歌 和 亚马逊 等 大 公司 , 我 不 太 担 心 这 些 问题 。 如 果 你 在 简历 上 提 到 自己 熟悉 某 
种 语言 , 那 你 自然 应 该 掌握 这 种 语言 的 基本 概念 。 不 过 , 我 还 是 建议 你 在 数据 结构 和 算法 方面 多 
下 工夫 。 
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参加 小 公司 和 非 软件 公司 的 面试 ， 这 些 问 题 可 能 更 显 重要 。 在 CareerCup.com 上 搜索 你 心仪 
的 公司 再 作 决 定 。 如 果 找 不 到 那 家 公司 ， 那 就 找 一 家 类 似 的 公司 作为 参照 。 一 般 而 言 ,创业 公司 
更 看 重 与 他 们 使 用 的 编程 语言 相关 的 技能 。 


6.2 如何 应 对 


面试 绝 非 易 事 。 要 是 没 能 立刻 答 出 所 有 问题 或 某 个 问题 , 也 没关系 ! 实际 上 , 根据 我 的 经 验 ， 
在 我 面试 过 的 120 多 人 中 ， 大 概 只 有 10 个 人 能 立即 答 上 我 经 常 问 的 问题 。 

因此 ， 碰 到 为 手 的 问题 ， 不 要 慌张 。 只 管 大 声 说 出 你 准备 怎么 解决 。 向 面试 官 说 明 你 会 如 何 
处 理 这 个 问题 ， 这 样 面试 官 就 不 会 误 以 为 你 被 难 住 了 。 

另外 , 还 有 一 点 : 只 有 面试 官 点 头 认可 ， 你 才 算 是 真正 解决 了 问题 ! 我 指 的 是 ， 在 给 出 算法 
后 ， 你 就 要 开始 考虑 它 可 能 存在 的 问题 。 边 写 代 码 ， 边 查 缺 陷 。 如 果 你 和 我 面试 过 的 其 他 110 名 
求职 者 一 样 ， 那 就 免不了 要 犯 一 些 错误 。 

解决 技术 面试 题 的 五 步 法 

解决 技术 面试 题 可 采取 下 面 的 五 步 法 。 

(1) 向 面试 官 提问 ， 以 消除 疑义 。 

(2) 设计 一 种 算法 。 

(3) 先 写 伪 码 ， 但 务必 告诉 面试 官 接 下 来 会 写 “ 真 实 的 ”代码 。 

(4) 写 代 码 要 不 紧 不 慢 。 

(5) 测试 写 好 的 代码 ， 仔 细 修 正 每 一 处 错误 。 

下 面 我 们 将 逐一 探讨 上 述 五 个 步骤 。 

第 一 步 : 提问 

技术 面试 题 看 似 清 晰 明确 实则 模糊 不 清 , 因此 务必 多 提问 题 以 澄清 所 有 存疑 之 处 。 问 到 最 后 ， 
你 可 能 会 发 现 ， 这 个 问题 与 你 最 初 预 想 的 截然 不 同一 一 也 许 更 难 ， 也 许 更 简单 。 实 际 上 , 许多 面 
试 官 (尤其 是 微软 的 ) 会 特意 考察 你 能 否 提出 好 问题 。 

好 问题 大 概 是 这 样 的 : 数据 类 型 是 什么 ? 有 多 少数 据 ? 解决 这 个 问题 需要 什么 假定 条 件 ? 用 








户 都 是 谁 ? 









































示例 :“ 设 计 一 种 列表 排序 算法 。 





口 问题 : 
回答 : 
口 问题 : 
回答 : 
口 问题 : 
回答 : 
口 问题 : 
回答 : 








具体 是 哪 种 列表 ? 数组 还 是 链表 ? 

数组 。 

数组 里 存放 的 是 什么 ? 数字、 字符、 还 是 字符 串 ? 
数字 。 

这 些 数字 都 是 整数 吗 ? 

是 的 。 

这 些 数字 来 自 何 处 ? 是 身份 证 号 码 还 是 别 的 什么 数值 ? 
顾客 年 龄 。 
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口 问题 ， 总 共有 多 少 顾客 ? 

回答 : 大 概 一 百 万 。 

现在 我 们 要 解决 一 个 与 最 初 理解 很 不 一 样 的 问题 : 对 一 个 包含 一 百 万 个 整数 的 数组 进行 排 
序 ， 这 些 整数 在 0 到 130 ( 一 个 合理 的 最 高 年 龄 ) 之 间 。 该 怎么 解决 这 个 问题 呢 ? 只 需 创 建 一 个 包 
含 130 个 元 素 的 数组 ， 然 后 计算 每 一 个 元 素 出 现 的 次 数 。 

第 二 步 : 设计 算法 

算法 的 设计 可 能 会 很 难 , 不 过 下 一 节 的 “算法 题 的 五 种 解法 ”可 以 帮 上 大 忙 。 在 设计 算法 时 ， 
记得 问 问 自己 以 下 几 个 问题 : 

口 该 算法 的 空间 和 时 间 复 杂 度 如 何 ? 

口 碰 到 大 量 数据 会 怎么 样 ? 

口 你 的 设计 会 引发 其 他 问题 吗 ? 例如 ， 你 设计 了 一 种 二 又 查找 树 的 变 体 ， 那 么 该 设计 是 

会 影响 插入 、 查 找 或 删除 时 间 ? 

口 如 果 还 有 其 他 问题 或 限制 ， 你 会 做 出 正确 的 取舍 吗 ” 对 于 哪些 场景 ， 这 一 取舍 可 能 不 是 

最 优 的 ? 

口 如 果 面 试 官 指定 特定 数据 (例如, 前面 提 到 待 处 理 数 据 是 年 龄 值 , 或 按 一 定 顺序 排列 的 )， 
你 能 否 善 用 该 信息 ”面试 官 给 你 特定 信息 往往 是 有 原因 的 。 

先 给 出 蛮 力 解法 ， 这 么 做 当然 是 允许 的 ,甚至 推荐 这 么 做 。 然 后 ,在 此 基础 上 不 断 优 化 。 很 
显然 , 面试 官 总 是 期 望 你 能 给 出 尽 可 能 最 优 的 解法 , 但 这 并 不 意味 着 一 开始 就 得 给 出 完美 无 瑕 的 
答案 。 

第 三 步 : 编写 伪 码 

先 写 伪 码 有 助 于 你 理 清 思 路 , 减少 犯错 的 次 数 。 不过, 务必 先 跟 面试 官 打 声 招呼 ， 你 会 先 写 
伪 码 ， 紧 接着 就 会 编写 “真实 的 ”代码 。 许 多 求职 者 选择 写 伪 码 ， 意 在 “逃避 ”编写 真实 代码 ， 
你 肯定 不 愿 与 那些 求职 者 为 伍 。 

第 四 步 : 编写 代码 

编写 代码 不 要 太仓 促 ; 实际 上 ， 太 仓促 很 可 能 会 害 了 你 自己 。 写 代码 时 只 管 放 松 步 调 ， 做 到 
有 条 不 率 ， 丝 丝 人 扣 。 另 外 ， 切 记 以 下 忠告 。 

口 多 用 数据 结构 : 根据 实际 情况 选用 合适 的 数据 结构 ， 或 者 自己 定义 数据 结构 。 例 如 ， 有 

个 面试 题 涉及 从 一 群 人 中 找 出 年 龄 最 小 的 ， 不 妨 考虑 定义 一 个 数据 结构 Person 表 示 一 个 
人 。 这 样 也 能 展现 出 你 注重 良好 的 面向 对 象 设计 。 
口 写 代码 不 要 太 杂 乱 : 这 看 似 小 事 一 桩 ， 实 则 很 重要 。 在 白板 上 写 代码 时 ， 尽 量 从 左上 角 
而 不 是 中 间 开 始 写 。 这 样 才 有 足够 的 地 方 从 容 答题 。 
第 五 步 : 测试 
没 错 ， 自 己 写 的 代码 自己 测试 ! 考虑 测试 以 下 用 例 。 
口 极端 用 例 : 0、 负 数 、 空 值 (null )、 最 大 值 、 最 小 值 。 
口 用 户 错 误 : 用 户 传人 空 值 或 负数 会 出 什么 问题 ? 
口 一 般 用 例 : 测试 正常 用 例 。 
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如 果 你 的 算法 很 复杂 或 涉及 大 量 数 值 操作 ( 移 位 、 算 术 运 算 等 )， 建 议 边 写 代 码 边 测试 ， 而 
不 是 写 完 代码 再 测试 。 

发 现 错误 时 (这 是 难免 的 )， 务 必 先 弄 清楚 出 现 缺陷 的 原因 再 作 修改 。 你 肯定 不 希望 自己 在 
面试 官 看 来 像 只 热 锅 上 的 蚂蚁 一 样 团团 乱 转 ,这 里 修 修 那 里 补 补 。 举 个 例子 ,我 就 碰 到 过 这 样 的 
求职 者 ， 他 发 现 函数 磁 到 某 个 特定 值 返 回 true 而 非 正确 的 false， 于 是 直接 修改 返回 值 ， 接 着 验 
证 函数 能 否 工作 。 这 或 许可 以 修正 那 种 特定 情况 下 出 现 的 问题 ， 但 无 疑 又 会 滋生 新 的 问题 。 

当 你 察觉 代码 中 存在 的 问题 时 ,务必 三 思 而 后 改 ， 先 理 清 代码 失效 的 原因 。 这 样 你 才能 写 出 
既 漂亮 又 整洁 的 代码 ， 也 会 越 写 越 快 。 


6.3 ”算法 题 的 五 种 解法 


要 解决 棘手 的 算法 问题 ， 世 上 没有 什么 不 二 法 门 ,不 过 下 面 介绍 的 几 种 方法 可 能 管用 。 常 计 
道 熟 能 生 巧 ， 题 目 练习 得 越 多 ， 就 越 容易 确定 该 采用 哪 种 方法 来 解决 问题 。 

另外 ,下面 这 五 种 方法 可 以 “混搭 ”使 用 。 也 就 是 说 , 施 以 “简化 推广 法 ”后 ， 还 可 以 接着 
尝试 “模式 匹配 法 ”。 

方法 一 :举例 法 

我 们 先 从 你 可 能 熟悉 的 “举例 法 ”开始 ， 也 许 你 从 未 听 过 这 种 叫 法 。“ 举 例 法 ”是 先 列举 一 
些 具体 的 例子 ， 看 看 能 否 发 现 其 中 的 一 般 规则 。 

示例 ， 给 定 一 个 具体 时 间 ， 计 算 时 针 与 分 针 之 间 的 角度 。 

下 面 以 3 点 27 分 为 例 。 确 定 3 点 的 时 针 位 置 和 27 分 的 分 针 位 置 ， 我 们 可 以 画 出 一 个 时 钟 。 

在 下 面 的 解法 中 ，/ 表 示 小 时 ，m 表 示 分 钟 。 同 时 ， 我 们 假定 /的 范围 是 0 一 23。 
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从 这 些 例子 可 以 得 出 以 下 规则 。 

口 分 针 的 角度 ( 从 12 点 整 开 始 算 起 ); 360 x m /60; 

口 时 针 的 角度 ( 从 12 点 整 开 始 算 起 ): 360 x (4 % 12)/12+360 x (m/60) x (1/12); 

口 时 针 和 分 针 之 间 的 角度 : (时 针 的 角度 -分 针 的 角度 ) % 360。 

简化 上 述 式 子 可 以 得 到 (30h -5.5m) % 360。 

方法 二 : 模式 匹配 法 

模式 匹配 法 是 指 将 现 有 问题 与 相似 问题 作 类 比 , 看 看 能 否 通过 修改 相关 问题 的 解法 来 解决 新 











示例 : 一 个 有 序数 组 的 元 素 经 过 循环 移动 ， 元 素 的 顺序 可 能 变 为 “3456712”。 怎样 才 能 
找 出 数组 中 最 小 的 那个 元 素 ? 假设 数组 中 的 元 素 各 不 相同 。 
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这 个 问题 和 下 面 两 个 问题 有 点 类 似 : 
口 在 一 个 无 序数 组 中 找 出 最 小 的 元 素 ; 
口 在 一 个 有 序数 组 中 找 出 某 个 特定 的 元 素 ( 比如 ， 通 过 二 分 查找 法 )。 
@ 处 理 方 法 
在 无 序数 组 中 查找 最 小 元 素 的 算法 没 多 大 意思 ( 只 要 遍历 所 有 元 素 即 可 )， 同 时 它 也 没有 利 
用 给 定 信息 〈( 即 这 是 一 个 有 序数 组 )， 因 此 这 个 问题 帮 不 上 什么 忙 。 
然而 ， 二 分 查找 法 就 非常 适合 。 我 们 知道 ， 这 是 个 有 序数 组 ， 只 是 一 部 分 元 素 循环 移动 过 。 
因此 元 素 排序 肯定 是 从 小 到 大 , 在 某 一 位 置 突然 变 小 , 接着 又 开始 从 小 到 大 排列 。 那 个“ 转折点” 
































正 是 最 小 的 元 素 。 
比较 中 间 元 素 与 末尾 元 素 (6 和 2 )， 由 于 MID > RIGHT， 可 以 确定 这 个 转折 点 就 在 这 两 个 元 
素 之 间 。 这 不 符合 从 小 到 大 的 排列 顺序 ， 故 而 表明 转折 点 就 在 其 中 。 





如 果 MID 比 RIGHT 小 ， 则 说 明 转 折 点 要 么 在 前 半 部 分 ， 要 么 根本 不 存在 〈 此 数组 严格 按照 从 
小 到 大 排序 )。 不 管 怎 样 ， 我 们 都 可 以 找到 最 小 的 元 素 。 

我 们 可 以 继续 运用 这 个 方法 ,将 数组 逐步 二 分 进行 查找 ,最终 找到 最 小 的 元 素 ( 或 是 转折 点 )。 

方法 三 : 简化 推广 法 

采用 简化 推广 法 , 我们 会 分 多 步 走 。 首 先 ,我们 会 修改 某 个 约束 条 件 ， 比 如 数据 类 型 或 数据 
量 ,， 从 而 简化 这 个 问题 。 接 着 ,我 们 转 而 处 理 这 个 问题 的 简化 版 本 。 最 后 ,一 旦 找到 解决 简化 版 
问题 的 算法 , 我 们 就 可 以 基于 这 个 问题 进行 推广 ,并 试 着 调整 简化 版 问题 的 解决 方案 , 让 它 适用 
于 这 个 问题 的 复杂 版 本 。 

示例 : 从 一 本 杂志 里 剪 下 一 些 单词 可 以 拼凑 成 一 封 勒索 信 。 怎 样 才能 断定 勒索 信 ( 以 字符 串 
表示 ) 是 否 由 某 本 杂志 ( 即 另 一 个 字符 串 ) 里 的 单词 组 成 ? 

我 们 可 以 先 这 样 简化 问题 : 暂时 不 考虑 单词 ， 只 当 它 是 字符 。 也 就 是 说 , 假设 我 们 从 杂志 
前 下 一 些 字符 拼 成 了 这 封 勒 索 信 。 

接着 , 我 们 只 需 新 建 一 个 数组 并 数 出 字符 的 数量 ， 即 可 解决 这 个 简化 后 的 勒索 信 问 题 。 数 组 
中 的 每 个 元 素 对 应 一 个 字母 。 首 先 , 我 们 数 出 每 个 字符 在 勒索 信 中 出 现 的 次 数 , 然后 再 遍历 整 本 
杂志 ， 确 认 它 是 否 包 含 勒索 信 上 的 全 部 字符 。 

推广 这 个 算法 时 ,具体 做 法 和 上 面 的 差不多 。 只 不 过 这 一 回 , 我 们 不 再 创建 包含 字符 计数 的 
数组 ， 而 是 创建 一 个 散 列 表 ， 将 单词 映射 到 其 词 频 上 。 

方法 四 : 简单 构造 法 

对 于 某 些 类 型 的 问题 , 简单 构造 法 非常 奏效 。 使 用 简单 构造 法 , 我 们 会 先 从 最 基本 的 情况 ( 比 
如 n = 1 ) 来 解决 问题 ， 一 般 只 需 记 下 正确 的 结果 。 得 到 n = 1 的 结果 后 ， 接 着 设法 解决 n = 2 的 情 
况 。 接 下 来 ， 有 了 n= 1 和 n= 2 的 结果 ， 我 们 就 可 以 试 着 解决 n = 3 的 情况 了 。 

最 后 ， 你 会 发 现 这 其 实 就 是 一 种 递归 算法 一 一 知道 N-1 时 的 正确 结果 ， 就 能 计算 出 N 时 的 结 
果 。 有 了 时， 只 有 等 到 算出 N 为 3 或 4 时 的 结果 ， 我 们 才能 从 中 找到 规律 ， 基 于 前 面 的 结果 解决 整个 


问题 。 
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示例 : 设计 一 种 算法 ， 打 印 某 个 字符 串 所 有 可 能 的 排列 组 合 。 为 简单 起 见 ， 假 设 字符 串 中 没 
有 重复 字符 。 

以 字符 串 abcdefg 为 例 : 

只 有 “a” 的 情况 ， 结果 为 : {“a”} 

然后 是 “ab”， 结果 为 : {“ab”，“ba”} 

再 然后 是 “abc”， 结 果 会 是 什么 呢 ? 

此 时 ， 问 题 开 始 变 得 “有 点 意思 ”了 。 得 到 P(“ab”) 的 答案 ， 怎 么 才能 生成 P(“abc”) 呢 ? 很 
简单 ， 新 字符 是 *c”， 我 们 只 需 在 前 一 种 情况 的 答案 也 即 字符 组 合 的 任意 位 置 加 一 个 < 就 可 以 了 。 
也 就 是 : 

p(“abc”) = 将 “c* 字 符 插入 P(“ab”) 得 到 的 所 有 字符 串 的 任意 位 置 。 

亦 即 : P(“abc”) = 将 “c” 字 符 插入 {“ab”，“ba”} 这 两 个 字符 串 中 的 任意 位 置 。 

也 就 是 : P(“abc”) = merge({“cab”, “acb”, “abc”}, {“cba”, “bca”, “bac”})。 

最 后 得 出 结果 : P(“abc”) = {“cab”,， “acb”, “abc”, “cba”, “bca”, “bac”}。 

既然 掌握 了 其 中 的 套路 , 我 们 就 可 以 设计 一 个 递归 算法 。 要 生成 字符 串 s1. . .sn 的 所 有 排列 ， 
我 们 可 以 先 “ 砍 掉 ” 最 后 一 个 字符 ， 首 先生 成 s1.. .sn 的 所 有 排列 。 得 到 sl. . .s m1 所 有 排列 的 
结果 列表 之 后 ， 我 们 会 循环 遍历 这 个 列表 ， 并 在 每 个 字符 串 的 任意 位 置 插入 sn。 

简单 构造 法 最 后 往往 会 演变 成 递归 法 。 

方法 五 :数据 结构 头脑 风暴 法 

这 种 方法 看 起 来 有 点 条 , 不 过 很 管用 。 我 们 可 以 快速 过 一 遍 数 据 结构 的 列表 ,然后 逐一 尝试 
各 种 数据 结构 。 这 种 方法 很 实用 ， 因 为 一 旦 找到 合适 的 数据 结构 ( 比如 说 树 )， 很 多 问题 也 就 迎 
刃 而 解 了 。 


示例 : 随机 生成 一 些 数字 ， 并 保存 到 一 个 ( 可 扩展 的 ) 数组 中 。 如 何 跟踪 数组 的 中 位 数 ? 


数据 结构 头脑 风暴 法 的 过 程 大 致 如 下 。 

口 链表 ? 念 怕 不 行 一 一 在 数字 的 存 取 和 排序 上 ， 链 表 往 往 效 果 不 佳 。 

口 数组 ? 也 许可 以 ， 不 过 你 已 经 用 了 一 个 数组 。 你 有 办 法 让 数组 保持 有 序 状态 吗 ? 这 人 么 做 

开销 候 怕 比较 大 。 暂 不 考虑 采用 ， 必 要 的 话 ， 可 以 回头 再 试 。 

口 二 义 树 ? 倒 也 有 可 能 ， 因 为 二 义 树 非常 适合 处 理 排序 问题 。 实 际 上 ， 如 果 这 棵 二 又 树 是 
完全 平衡 的 ， 根 结 点 可 能 就 是 中 位 数 。 不 过 ， 你 要 小 心 一 一 如 果 它 包含 偶数 个 元 素 ， 那 
么 中 位 数 实际 上 是 中 间 两 个 元 素 的 平均 值 。 而 中 间 两 个 元 素 不 可 能 都 是 根 结 点 。 因 此 ， 
二 又 树 也 许可 行 ， 我 们 待 会 再 说 。 

口 堆 ? 堆 非 常 适合 基本 排序 ,跟踪 最 大 值 和 最 小 值 。 堆 其 实 也 很 有 意思 一 一 只 用 两 个 堆 ， 就 能 跟 
踪 较 大 的 那 一 半 元 素 和 较 小 的 那 一 半 元 素 。 较 大 的 一 半 保 存在 小 项 堆 ( min heap ) 中 ， 其 中 最 
小 元 素 位 于 堆 项 。 较 小 的 一 半 则 保存 在 大 顶 堆 (max heap ) 中 , 其 中 最 大 元 素 位 于 堆 项 。 现在 ， 
有 了 这 些 数据 结构 ， 整 个 数组 的 中 位 数 很 可 能 就 是 两 个 堆 顶 之 一 。 如 果 这 两 个 堆 大 小 不 一 样 ， 
你 可 以 从 元 素 较 多 的 堆 中 弹出 一 个 元 素 并 压 入 男 一 个 堆 中 ， 两 个 堆 很 快 就 能 “ 重 获 平衡 ”。 
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切记 ,问题 演练 得 越 多 ,你 就 越 容易 判断 该 选用 哪 种 数据 结构 。 当 然 了 ,你 也 能 更 自如 地 从 
这 五 种 方法 中 选 出 最 管用 的 那 种 。 


6.4 怎样 才 算 好 代码 


至 此 ,你 也 许 明白 了 , 许多 公司 都 想 找 能 写 出 “优美 、 整 洁 ” 代 码 的 人 才 。 但 这 到 底 意 味 着 
什么 ， 怎 样 才能 在 面试 中 展现 出 这 方面 的 能 力 呢 ? 
一 般 说 来 ， 好 代码 具备 如 下 特性 。 

口 正确 : 代码 应 当 正 确 处 理 所 有 预期 输入 (expected input ) 和 非法 输入 (unexpected input )。 

D 高 效 : 不 管 是 从 空间 上 还 是 从 时 间 上 来 衡量 ， 代 码 都 要 尽 可 能 地 高 效 运行 。 所 谓 的 “高 

效 ” 不 仅 是 指 在 极限 情况 下 的 渐 近 效率 ( asymptotic efficiency， 大 O 记 法 )， 同 时 也 包括 实 

际 运 行 的 效率 。 也 就 是 说 , 在 计算 O 时 间 时 , 你 可 以 忽略 某 个 常量 因子 , 但 在 实际 环境 中 ， 

该 常量 因子 可 能 有 很 大 影响 。 

简洁 : 代码 能 写成 10 行 就 不 要 写成 100 行 。 这 样 开 发 人 员 才 能 尽快 写 好 代码 。 

口 易 读 : 要 确保 其 他 开发 人 员 能 读 懂 你 的 代码 ， 并 弄 清 楚 来 龙 去 脉 。 易 读 的 代码 会 有 适当 
注释 ， 实 现 思路 也 简单 易 懂 。 这 就 意味 着 ,那些 包含 诸多 位 操作 的 花 俏 的 代码 不 见得 就 
是 “好 ”代码 。 

口 可 维护 : 在 产品 生命 周期 内 ， 代 码 经 过 适当 修改 就 能 应 对 需求 的 变化 。 此 外 ， 无 论 对 于 

原 开 发 人 员 还 是 其 他 开发 人 员 ， 代 码 都 应 该 易于 维护 。 

力求 实现 上 述 特性 必须 找到 一 个 平衡 点 。 比 如 ,有 些 情况 下 , 我 们 往往 要 牺牲 一 定 的 效率 好 
让 代码 更 易 维 护 ， 有 时 则 要 反 其 道行 之 。 

在 面试 中 ， 写 代码 时 应 该 好 好 考虑 这 些 要 素 。 下 文 就 前 面 的 清单 给 出 更 具体 的 描述 。 

1. 多 用 数据 结构 

假设 面试 官 要 求 你 编写 一 个 函数 ， 对 两 个 简单 的 多 项 式 求 和 ， 其 形式 为 4xe+ Bxs +… ( 其 中 
系数 和 指数 为 任意 正 实数 或 负 实数 ), 即 多 项 式 的 每 一 项 都 是 一 个 常量 乘 以 某 个 数 的 n 次 窜 。 面试 

官 还 补充 说 ， 不 必 对 这 些 多 项 式 做 字符 串 解析 ， 可 以 使 用 任意 数据 结构 来 表示 它们 。 

这 个 函数 有 多 种 实现 方式 。 

@ 最 差 的 实现 方式 

最 差 的 实现 方式 就 是 将 多 项 式 存储 为 一 个 double 型 数组 , 其 中 第 k 个 元 素 对 应 的 是 多 项 式 中 x 
的 系数 。 采 用 这 种 结构 有 一 定 问 题 ， 如 此 一 来 ， 多项式 就 不 能 含有 负 的 或 非 整 数 指数 。 要 想 用 这 
种 方法 来 表示 zx" 多 项 式 的 话 ， 这 个 数组 就 得 包含 1000 个 元 素 。 

1 int[] sum(double[] poly1，double[] poly2) { 

2 a 

@ 较 差 的 实现 方式 

一 种 不 算 最 差 的 实现 方式 是 将 多 项 式 存 为 一 对 数组 coefficients 和 exponents。 采用 这 种 方 
法 ， 多 项 式 的 所 有 项 可 以 按 任意 顺序 存放 ， 只 要 系数 和 指数 配对 ， 多 项 式 的 第 i 项 表示 为 

















口 
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。 。 。 ts[i 
Coefficients[i] * Xexponents[]。 


采用 这 种 实现 方式 ， 如 果 coefficients[p] = k 和 exponents[p] = m， 则 第 p 项 为 be"。 尽 
这 么 做 没有 上 面 那 种 解法 的 限制 , 但 还 是 很 凌乱 。 一 个 多 项 式 就 要 用 两 个 数组 记录 。 如 果 两 个 数 
组 长 度 不 同 ,多项式 就 会 出 现 “ 未 定义 ” 值 。 而 要 返回 多 项 式 更 是 麻烦 ， 因 为 一 下 子 得 返回 两 个 
数组 。 


天 














1 ?3?? sum(double[] coeffs1，double[] expon1， 

2 double[] coeffs2, double[] expon2) { 

3 

4 } 

® 较 好 的 实现 方式 

对 于 这 个 问题 ， 较 好 的 实现 方式 是 专 为 多 项 式 设计 一 种 数据 结构 。 





class PolyTerm { 
double coefficient; 
double exponent; 


} 


1 
2 
3 
4 
5 
6 PolyTerm[] sum(PolyTerm[] poly1，PolyTerm[] poly) { 
7 i 
8 } 

有 些 人 可 能 或 真 的 认为 这 么 做 “优化 过 了 头 ”。 也 许 是 , 也许 不 是 。 不管 你 是 不 是 这 么 认为 ， 
上 面 的 代码 都 表明 你 应 该 用 心思 考 如 何 设 计 代码 ， 而 不 要 匆忙 地 胡乱 堆砌 一 通 。 

2. 适当 重用 代码 

假设 面试 官 要 求 你 编写 一 个 函数 检查 某 个 二 进 制 数 ( 以 字符 串 形式 传人 ) 是 否 等 于 以 字符 中 
表示 的 十 六 进 制 数 。 

我 们 可 以 善 用 代码 重用 巧妙 解决 该 问题 。 





















































Ud 








1 public boolean compareBinToHex(String binary, String hex) { 
2 int n1 = convertToBase(binary, 2); 

3 int n2 = convertToBase(hex, 16); 

4 if (n1 <0 ||n2<0@)ft 

5 return false; 

6 } else { 

7 return n1 == n2; 

8 } 


9 } 


11 public int digitToValue(char c) { 
12 if (c >= “6 && c <= ‘9’) return c - ‘0’; 
13 else if (c >= ‘A’? && c = ‘F’) return 16 +c- A’; 


14 else if (c >= ‘a’ && c <= ‘f’) return 106 + Cc - ‘a’; 
15 return -1; 

16 } 

17 


18 public int convertToBase(String number, int base) { 
19 if (base < 2 || (base > 16 && base != 16)) return -1; 
20 int value = 0@; 
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21 for (int i = number.length() - 1; i >= 6; i--){ 


22 int digit = digitToValue(number.charAt(i)); 
23 if (digit < @ || digit >= base) { 

24 return -1; 

25 } 

26 int exp = number.length() - 1 - i; 

27 value += digit * Math.pow(base, exp); 

28 } 

29 return value; 

36 } 

















我 们 本 可 以 实现 两 套 代码 , 实现 二 进 制 数 和 十 六 进 制 数 的 转换 , 但 这 么 做 只 会 加 大 代码 的 编 
写 难 度 , 而 且 维 护 起 来 也 更 难 。 相 反 ， 我 们 还 是 通过 编写 convertToBase 和 digitTovalue 的 方法 
来 重用 代码 。 

3. 模块 化 
编写 模块 化 代码 是 指 将 孤立 的 代码 块 划 分 为 相应 的 方法 ( 函数 )。 这 有 助 于 让 代码 更 易 读 ， 
可 读 性 和 可 测试 性 更 强 。 

假设 你 在 编写 交换 整数 数组 中 的 最 大 和 最 小 元 素 的 代码 ， 不 妨 将 全 部 代码 写 在 一 个 函数 里 ， 
如 下 所 示 : 















































1 public void swapMinMax(int[] array) { 

2 int minIndex = 0@; 

3 for (int i = 1; i < array.length; i++) { 
4 if (array[i] < array[minIndex]) { 

5 minIndex = i; 

6 

7 

8 


} 

} 
9 int maxIndex = 0@; 
16 for (int i = 1; i < array.length; i++) { 
11 if (array[i] > array[maxIndex]) { 
12 maxIndex = i; 
13 } 
14 } 
15 
16 int temp = array[minIndex]; 
17 array[minIndex] = array[maxIndex]; 
18 array[maxIndex] = temp; 
19 } 


或 者 ， 你 还 可 以 采取 更 模块 化 的 方式 ， 将 相对 孤立 的 代码 块 隔离 到 对 应 的 方法 中 。 


1 public static int getMinIndex(int[] array) { 
2 int minIndex = 0@; 

3 for (int i = 1; i < array.length; i++) { 
4 if (array[i] < array[minIndex]) { 

5 minIndex = i; 

6 } 

7 } 

8 return minIndex; 

9 } 
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16 

11 public static int getMaxIndex(int[] array) { 
12 int maxIndex = 0@; 

13 for (int i = 1; i < array.length; i++) { 
14 if (array[i] > array[maxIndex]) { 

15 maxIndex = i; 

16 } 

17 } 

18 return maxIndex; 

19 } 

20 


21 public static void swap(int[] array, int m, int n) { 
22 int temp = array[m]; 


23 array[m] = array[n]; 

24 array[n] = temp; 

25 } 

26 

27 public static void swapMinMaxBetter(int[] array) { 
28 int minIndex = getMinIndex(array); 

29 int maxIndex = getMaxIndex(array); 

36 swap(array, minIndex, maxIndex); 

31 } 

















虽然 前 面 的 非 模块 化 代码 看 起 来 也 不 怎么 糟 , 但 模块 化 代码 的 一 大 好 处 在 于 它 易于 测试 ， 
为 每 一 部 分 都 可 以 单独 验证 。 随 着 代码 越 来 越 复杂 ,编写 模块 化 代码 就 变 得 越发 重要 。 模块 化 的 
代码 也 更 易 阅 读 和 维护 。 面 试 官 希望 看 到 你 能 在 面试 中 展现 这 些 技能 。 

4. 灵活 、 健 壮 

不 要 因为 面试 官 要 求 编写 代码 检查 谁 是 三 连 棋 游 戏 的 赢家 , 就 非得 假定 它 是 一 个 3x3 的 棋盘 。 
何不 放手 针对 NxN 模 盘 编 写 代 码 呢 ? 

编写 灵活 、 通 用 的 代码 , 也 可 能 意味 着 使 用 变量 ， 而 不 是 在 代码 里 直接 把 值 写 死 ， 或 者 使 用 
模板 / 泛 型 来 解决 问题 。 要 是 有 办 法 编写 代码 解决 更 普遍 的 问题 ， 那 我 们 就 应 该 这 么 做 。 

当然 , 它 也 有 限制 条 件 。 如 果 通 用 解决 方案 更 为 复杂 ,并 且 在 面试 中 几乎 没有 必要 使 用 ， 那 
就 按照 要 求解 决 相应 的 问题 ， 效 果 可 能 会 更 好 。 

5. 错误 检查 

写 代 码 很 细心 的 人 有 一 个 明显 的 特征 ,， 那 就 是 她 不 会 想当然 地 处 理 输入 信息 。 相 反 ,， 她 会 用 
ASSERT 语 句 或 if 语句 仔细 验证 输入 数据 是 否 合理 。 

比如 ， 回 到 前 面 那 段 将 基数 为 ;的 进 制 数 ( 比如 基数 为 2 或 16 ) 转换 成 整数 的 代码 。 



































1 public int convertToBase(String number, int base) { 

2 if (base < 2 || (base > 16 && base != 16)) return -1; 
3 int value = 0; 

4 for (int i = number.length() - 1; i >= 6j i--) { 

5 int digit = digitToValue(number.charAt(i)); 

6 if (digit < 6 || digit >= base) { 

7 return -1; 

8 } 

9 int exp = number.length() - 1 - i; 

16 value += digit * Math.pow(base, exp); 
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证 } 
12 return value; 
13 } 








在 第 2 行 ， 我 们 检查 基数 是 否 有 效 ( 假定 除 16 外 ， 大 于 10 的 基数 都 是 无 效 的 ， 没 有 标准 的 字 
符 串 表 示 形 式 )。 在 第 6 行 ， 我们 男 加 了 一 处 错误 检查 : 确保 每 个 数字 都 落 在 允许 范围 内 。 

诸如 此 类 的 错误 检查 在 实际 的 产品 代码 中 至 关 重 要 ， 因 此 ， 面 试 中 也 不 能 掉以轻心 。 

当然 , 这 些 错误 检查 有 时 很 繁琐 ,可 能 会 浪费 宝贵 的 面试 时 间 。 关 键 在 于 指出 你 会 加 上 错误 
检查 。 如 果 错 误 检 查 远 非 一 条 if 洛 句 就 能 搞定 ， 写 代码 时 最 好 先 为 错误 检查 预 留 一 些 空间 ， 并 告 
诉 面试 官 ， 完 成 其 余 代码 之 后 你 会 补 上 错误 检查 代码 。 
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录用 通知 及 其 他 


口 如 何 处 理 录用 和 被 拒 的 情况 
口 如 何 评 估 录 用 待遇 

口 录用 谈判 

口 人 职 须知 


7.1 如 何 处 理 录用 与 被 拒 的 情 ; 


面试 结束 后 ， 刚 觉得 可 以 松口 气 了 ， 你 可 能 义 会 陷入 “面试 后 综合 征 ”: 要 接受 这 家 公司 的 
录用 吗 ? 它 是 理想 之 选 吗 ?” 如 何 拒绝 录用 通知 ?怎么 处 置 回复 期 限 ? 我 们 先 来 探讨 这 些 问 题 , 接 
下 来 几 节 会 细 说 如 何 评估 录用 待遇 ， 以 及 该 怎样 讨价还价 。 

1. 回复 期 限 与 延长 期 限 

录用 通知 大 都 附 有 回复 期 限 , 一般 为 一 到 四 周 。 不过, 要 是 还 在 苦 等 其 他 公司 的 回音 ,你 可 
以 请 求 发 出 录用 通知 的 公司 延长 回复 期 限 。 条 件 允 许 的 话 , 大 部 分 公司 都 会 通 情 达 理 , 予以 配合 。 

2. 如 何 拒绝 录用 通知 

拒绝 公司 的 录用 通知 很 讲究 技巧 。 即 使 你 现在 对 该 公司 不 感 兴趣 ， 没 准 几 年 后 又 感 兴趣 了 。 
又 或 者 , 该 公司 与 你 打 过 交道 的 联系 人 跳 到 为 一 家 更 令 人 心动 的 公司 。 因 此 , 你 最 好 还 是 礼貌 得 
体 地 拒绝 录用 通知 ， 并 与 该 公司 做 好 沟通 。 

拒绝 录用 通知 时 ， 请 给 出 一 个 合乎 情理 旦 不 容 置疑 的 理由 。 比 如 , 若 要 舍 大 公司 而 取 创 业 公 
司 , 你 可 以 阐明 自 认为 创业 公司 是 当下 最 佳 选择 的 理由 。 这 两 种 公司 截然 不 同 ,大 公司 也 不 可 能 
突然 变 成 创业 公司 ， 所 以 大 公司 对 此 也 无 可 厚 非 。 

3. 处 理 被 拒 的 情况 

科技 巨头 公司 一 般 会 拒 掉 大 约 80% 的 求职 者 , 但 他 们 也 明白 ,这些 面试 未 必 能 充分 考察 求职 
者 的 能 力 。 有 鉴于 此 ,他 们 通常 会 给 之 前 被 拒 的 求职 者 再 次 面试 的 机 会 。 甚 至 有 些 公司 会 主动 联 
系 以 前 的 求职 者 ,或 是 因为 求职 者 的 面试 表现 而 加 快 处 理 流程 。 

当 你 接 到 拒 电 时 , 把 它 视 作 一 时 的 挫折 而 非 终 身 裁决 。 礼 貌 地 感谢 招聘 人 员 为 此 付出 的 时 间 
和 精力 ， 表 达 自 己 的 遗憾 之 情 ， 对 他 们 的 决定 表示 理解 ， 并 询问 什么 时 候 可 以 再 次 申请 。 

找 出 被 拒 的 原因 很 难 , 招聘 人 员 一 般 也 不 会 吐露 实情 。 当 然 ， 如 果 你 拐弯 抹 角 地 询问 下 次 面 
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试 该 注意 哪些 事项 , 运气 好 的 话 , 说 不 定 可 以 打探 到 其 中 的 缘由 。 你 也 可 以 回想 一 下 自己 在 面试 
中 的 表现 , 但 根据 我 的 经 验 , 求职 者 一 般 无 法 作出 准确 分 析 。 你 可 能 认为 是 因为 自己 解决 某 个 问 
题 时 大 费 周 折 , 不 过 这 都 是 相对 的 ; 你 并 不 清楚 自己 的 解 题 节奏 比 其 他 求职 者 是 快 还 是 慢 ? 实际 
上 ,一般 来 说 , 求职 者 被 拒 主要 是 因为 编程 与 算法 功底 不 过 关 ， 总 之 你 要 在 这 些 方面 狠 下 工夫 。 


7.2 如何 评估 录用 待遇 


恭喜 你 ! 拿 到 录用 通知 了 ! 幸运 的 话 , 你 可 能 手 握 不 止 一 个 录用 通知 。 现 在 ,招聘 人 员 的 工 
作 就 是 尽 其 所 能 说 服 你 签约 。 那么 , 又 该 怎么 判断 这 家 公司 是 和 否 适合 自己 呢 ? 下 面 我 们 将 逐一 探 
讨 评估 录用 待遇 的 若干 注意 事项 。 

1. 薪酬 待遇 的 考量 

在 评估 录用 通知 时 , 求职 者 可 能 会 犯 的 最 大 错误 也 许 就 是 过 于 看 重 薪 水 。 如 此 一 叶 障 目 导 致 

有 些 求 职 者 最 后 反而 接受 了 一 个 更 差 的 录用 通知 。 薪 水 只 是 薪酬 待遇 的 一 部 分 。 你 还 应 考虑 以 下 

儿 点 s 

口 签约 奖金 、 搬 家 费 及 其 他 一 次 性 津贴 : 很 多 公司 都 会 提供 签约 奖金 ， 有 的 还 会 给 搬家 费 。 

在 比较 待遇 时 ， 最 好 将 这 些 一 次 性 津贴 除 以 3 (或 者 你 预期 服务 的 年 限 )。 

口 各 地 生活 成 本 差异 : 收 到 多 个 来 自 不 同 地 区 的 录用 通知 ， 不 要 小 看 地 域 差别 带 来 的 影响 。 
比如 ， 硅 谷 的 生活 成 本 就 比 西雅图 要 高 出 约 20% 至 30%， 其 中 部 分 原因 是 加 州 要 收 10% 的 
州 税 ， 华 盛 顿 州 则 不 用 。 你 可 以 找 几 个 相关 网 站 来 估算 各 地 的 生活 成 本 。 

口 年 终 奖 : 科技 公司 的 年 终 奖 大 约 在 3% 到 30% 之 间 。 招聘 人 员 可 能 会 告知 年 终 奖 的 平均 数 ， 

没有 的 话 ， 不 妨 找 公司 里 的 朋友 打听 。 

口 股票 期 权 与 补助 金 : 这 部 分 收入 也 可 能 是 全 年 收入 的 男 一 大 块 。 就 像 签 约 奖金 一 样 ， 你 
也 可 以 将 这 部 分 收入 除 以 3， 然 后 把 该 数 日 计 和 年薪。 

当然 , 切记 一 点 , 能 学 到 的 知识 及 公司 对 你 职业 生涯 的 影响 远 比 薪水 来 得 重要 。 务 请 慎重 考 

虑 当下 薪资 对 你 到 底 有 多 重要 。 

2. 职业 发 展 
尽管 收 到 录用 通知 是 如 此 令 人 兴奋 , 甚至 有 时 候 幸福 感 还 能 持续 上 几 年 , 但 同时 你 应 该 开始 

考虑 未 来 的 职业 发 展 方向 。 因 此 , 现在 就 思考 这 份 工作 会 对 你 的 职业 发 展 有 怎样 的 影响 ,非常 重 

要 。 也 就 是 ， 要 关注 下 列 问 题 : 

口 该 公司 名 号 能 否 增加 自身 履历 的 份量 ? 

口 我 能 学 到 多 少 知识 ”我 会 学 到 相关 领域 的 技术 吗 ? 

口 该 职位 有 无 升迁 可 能 ” 开发 人 员 的 职业 路 径 是 什么 样 的 ? 

口 想 转 到 管理 岗位 的 话 ， 该 公司 是 否 提供 了 切实 可 行 的 通道 ? 

口 该 公司 或 团队 是 否 处 于 上 升 期 ? 

口 想 要 跳槽 的 话 ， 该 公司 所 在 地 是 否 有 很 多 其 他 机 会 ? 我 需要 搬家 吗 ? 

最 后 一 点 非常 重要 ， 也 很 容易 被 人 忽视 。 如 果 你 在 微软 硅谷 分 部 工作 ， 跳 槽 时 会 有 许多 机 
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会 。 然 而 ， 要 是 在 微软 西雅图 总 部 ， 选 择 余地 只 剩 下 亚马逊 、 谷 歌 和 其 他 一 些小 公司 。 此 外 ， 
要 是 去 了 上 弗吉尼亚 州 杜 勒 斯 的 AOL， 那 选择 余地 就 更 小 。 所 以 , 千 万 不 要 忽视 地 理 位 置 这 个 因 
素 ， 否 则 你 可 能 被 迫 在 某 家 公司 “终老 ”"， 只 因为 那里 没 别 的 公司 可 去 ， 除 非 完 全 改变 自己 的 
生活 方式 。 

3. 公司 稳定 与 否 

每 个 人 的 境遇 都 有 所 不 同 , 不 过 , 我 一 般 都 会 鼓励 求职 者 不 要 太 在 意 公司 稳定 与 否 。 真 要 碰 
上 了 裁员 , 那 你 肯定 也 能 在 同类 公司 找到 一 方 新 天 地 。 你 要 确认 的 问题 是 : 要 是 被 解雇 了 ， 你 会 
怎么 办 ?你 对 找到 新 工作 是 否 信心 满 满 ? 

4. 幸福 指数 

当然 ， 幸 福 指 数 也 是 一 个 重要 考量 。 以 下 因素 都 会 影响 你 工作 的 幸福 感 。 
口 产品 : 很 多 人 都 非常 看 重 自己 做 的 产品 ， 当 然 这 也 是 一 个 重要 方面 。 然 而 ， 对 大 多 数 工 
程 师 来 说 ， 还 有 比 这 更 重要 的 因素 ， 比 如 ， 与 哪些 人 一 起 共事 。 
口 经 理 与 队友 : 当 人 们 提 及 自己 热爱 或 痛恨 自己 的 工作 时 ， 通 常 是 他 们 的 队友 与 经 理 占 了 
主因 。 你 有 没有 跟 未 来 的 经 理 、 队 友 碰 过 面 ? 你 喜欢 和 他 们 交流 吗 ? 
口 企业 文化 : 企业 文化 涉及 方方面面 ， 从 如 何 作 决 策 到 整体 氛围 及 公司 的 组 织 架 构 。 不 妨 
问 问 未 来 的 同事 ， 看 看 他 们 会 如 何 描述 公司 的 企业 文化 。 
口 工作 时 长 : 问 一 问 未 来 的 队友 ， 他 们 一 般 工 作 多 长 时 间 ， 确 定 是 和 否 契 合 自己 的 生活 节奏 。 

不 过 ， 值 得 注意 的 是 ， 临 近 产 品 发 布 时 ， 加 班 在 所 难免 。 

此 外 ， 你 还 要 看 看 是 否 有 机 会 在 不 同 的 团队 轮 岗 〈 比如 在 谷歌 就 很 宽松 )， 万 一 不 喜欢 ， 你 

还 有 机 会 找到 更 合适 的 团队 和 部 门 。 














































































































7.3 录用 谈判 


2010 年 年 末 , 我 报 了 一 个 谈判 训练 班 。 第 一 天 ,培训 师 让 我 们 设想 一 个 购车 的 场景 。 经销 商 
A 报 的 是 一 口 价 ，2 万 美元 。 而 经 销 商 B 允 许 议 价 。 那 么 ， 要 讲 下 多 少 钱 你 才 愿 意 去 经 销 商 B 那 里 
买 车 呢 ?”【( 快 ! 迅速 报 出 你 的 答案 ! ) 

最 后 ,全 班 给 出 的 平均 数目 是 便宜 750 美 元 。 换 言 之 , 学 员 们 都 愿意 付 750 美 元 ,免除 一 小 时 
的 讨价还价 。 这 也 没什么 奇怪 的 ,在 对 全 班 学 员 进 行 的 民 调 中 , 大 部 分 人 都 表示 自己 接受 工作 录 
用 时 也 不 会 讨价还价 。 公 司 给 多 少 就 是 多 少 。 

拜托 ， 请 理直气壮 地 还 还 价 吧 。 下 面 是 几 点 可 资 参考 的 建议 。 

(1) 要 理直气壮 。 是 的 ， 迈 出 第 一 步 很 难 ， 没 什么 人 喜欢 谈判 。 但 讨价还价 还 是 有 必要 的 。 
招聘 人 员 不 会 因为 你 有 异议 就 撤回 录用 通知 ， 所 以 你 也 不 会 有 什么 损失 。 

(2) 最 好 手头 有 其 他 选择 。 从 根本 上 来 说 ,招聘 人 员 愿 意 与 你 谈判 是 因为 他 们 希望 你 能 加 入 
公司 。 如 果 你 手 涉 有 其 他 选择 ， 他 们 就 会 更 担心 你 有 可 能 拒绝 他 们 的 录用 邀约 。 

(3) 提出 具体 的 “要 价 "。 给 一 个 具体 的 数目 ， 比 如 要 求 年 薪 增 加 7 千 美 金 会 比 泛泛 地 要 求 涨 薪 
效果 更 佳 。 毕 浣 ， 如 果 只 是 要 求 涨 薪 ， 招 聘 人 员 可 以 不 痛 不 痒 地 加 个 1 千 块 来 打发 你 。 
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(4) 开 出 比 预期 稍 高 的 价 码 。 在 谈判 中 ， 人 们 一 般 不 会 全 盘 接 受 你 的 要 求 ， 总 是 要 讨价还价 
一 番 。 因 此 ， 你 开 的 价 码 可 以 比 自己 预期 的 高 一 些 ， 这 样 公司 再 往 下 降 一 降 ， 最 后 凤 大 欢喜 。 

(5) 不 要 只 时 着 薪水 。 公 司 更 愿意 就 薪水 之 外 的 条 件 作 出 让 步 ， 因 为 给 你 大 幅 涨 薪 可 能 会 造 
成 团队 内 部 同 工 不 同 酬 的 情况 。 你 可 以 稍 作 变 通 ， 要 求 更 多 的 期 权 或 签约 奖金 。 同 样 ， 还 可 以 要 
求 公司 将 搬家 费 直接 折算 成 现金 。 这 对 应 届 毕 业 生来 说 更 划算 ， 因 为 他 们 家 什 少 , 搬家 也 花 不 了 
多 少 钱 。 

(6) 使 用 最 合适 的 方法 。 很 多 人 会 建议 你 通过 电话 进行 谈判 。 在 一 定 程度 上 ， 他 们 是 对 的 。 
当然 , 要 是 不 喜欢 在 电话 中 讨价还价 ,可 以 使 用 电子 邮件 。 最 重要 的 是 你 本 人 有 谈判 的 想法 , 效 
果 比 形式 更 重要 。 

此 外 ， 与 大 公司 进行 谈判 ， 你 要 了 解 这 些 公司 都 有 某 种 职 等 级 别 制度 ， 一 定 的 级 别 对 应 一 定 的 
薪资 范围 。 微 软 对 此 就 有 明确 的 规定 。 你 可 以 在 对 应 范围 内 讨价还价 ， 但 要 价 太 高 就 会 超出 这 个 范 
围 。 如 果 你 觉得 自己 可 以 拿 到 更 高 级 别 , 那 就 得 向 招聘 人 员 和 未 来 的 团队 证 明 你 有 这 个 实力 一 一 
谈判 过 程 会 比较 难 ， 但 也 不 是 没有 可 能 。 


7.4 入 职 须知 


入 职 不 是 终点 , 而 是 你 职业 生涯 的 新 起 点 。 一 旦 正式 加 入 一 家 公司 ,你 就 得 开始 做 好 职业 规 
划 。 你 想 达到 什么 样 的 目标 ， 如 何 才能 实现 ? 

1. 制定 时 间 表 

“入 此 门 后 ， 非 疯 即 癫 ”， 这 种 情况 很 常见 。 新 生活 开始 之 际 总 是 很 美好 的 。 可 五 年 之 后 ,你 
还 停留 在 原 地 不 动 。 到 那 时 才 意 识 到 自己 虚度 了 最 近 三 年 的 时 光 , 技术 没什么 长 进 , 履历 也 乏 善 
可 陈 。 当 初 为 什么 不 待 上 两 年 就 走 呢 ? 

志 得 意 满 之 际 反而 是 最 危险 的 时 候 , 让 你 “温水 者 青蛙 ”而 忘记 了 百 尺 笔头 更 进一步 。 这 也 
正 是 工作 伊始 就 要 做 好 职业 规划 的 原因 。 好 好 想 一 想 ， 十 年 后 想 干 什么 ”该 如 何 一 步 步 达成 目 
标 ? 此外, 每 年 都 要 总 结 一 下 过 去 一 年 自己 在 职业 与 技能 上 取得 了 哪些 进步 , 明年 又 有 什么 样 的 
规划 ? 

提前 做 好 规划 并 定期 对 照 检 查 ， 这 样 ， 就 能 避免 自己 陷 人 “温水 者 青蛙 ”的 困境 。 

2. 打造 坚实 的 人 际 网 络 

在 找 新 工作 时 ， 人 际 网 络 的 作用 很 大 。 毕 竟 ,， 在线 申请 工作 有 很 多 不 确定 因素 ; 有 人 推荐 的 
话 就 会 好 很 多 ， 而 这 取决 于 你 的 关系 网 有 多 强大 。 

所 以 , 在 工作 中 要 与 经 理 、 同事 建 立 良 好 的 关系 。 就 算 有 人 离职 , 你 们 也 可 以 继续 保持 联系 。 
比如 ,在 他 们 离职 几 周 后 ， 写 封 简短 的 邮件 问候 一 下 ， 这 不 仅 可 以 拉 近 你 们 的 距离 ， 还 可 以 将 原 
本 的 同事 关系 升华 为 朋友 关系 。 

这 些小 技巧 同样 适用 于 你 的 个 人 生活 。 你 的 朋友 、 朋 友 的 朋友 都 是 你 的 宝贵 资源 。 我 为 人 人 ， 
人 人 为 我 。 
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3. 向 经 理 寻求 帮助 

有 些 经 理 很 愿意 提携 下 属 ， 帮 助 开拓 职业 道路 ,但 也 有 些 人 会 不 闻 不 问 。 所 以 ,这 都 要 看 你 
自己 是 否 有 心 开 拓 进 取 ， 寻 求 更 好 的 职业 发 展 。 

请 开 诚 布 公 地 向 你 的 主管 表明 心迹 。 如 欲 从 事 更 多 后 端 编程 项 目 , 不 妨 直言 相 告 。 如 要 往 管 
理 层 发 展 ， 你 可 以 与 经 理 探讨 自己 需要 做 些 什么 。 

记得 时 时 为 自己 打气 ， 这 样 才 能 逐步 实现 既定 目标 。 

















图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 


请 登录 我 们 的 网 站 www.CrackingTheCodingInterview.com， 下 载 完 整 可 编译 的 Java/Eclipse 工 
程 , 并 与 其 他 读者 一 起 讨论 书 中 的 面试 题 , 提交 问题 , 查看 本 书 勘误 , 发 布 简历 及 寻求 其 他 建议 。 
数据 结构 
口 数组 与 字符 串 
口 链表 
口 栈 与 队列 
口 树 与 图 
概念 与 算法 
口 位 操作 
口 智力 题 
口 数学 与 概率 
口 面向 对 象 设 计 
口 递归 和 动态 规划 
口 扩展 性 与 存储 限制 
口 排序 与 查找 
口 测试 
知识 类 问题 
口 C 和 C++ 
口 Java 
口 数据 库 
口 线程 与 锁 
附加 面试 题 


口 中 等 难题 
D 高 难度 题 
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8.1 数组 与 字符 串 


想必 本 书 读者 都 很 熟悉 什么 是 数组 和 字符 串 , 因此 这 里 不 再 歼 述 细节 。 我 们 会 把 重心 放 在 这 
些 数据 结构 相关 的 一 些 和 常见 技巧 和 问题 上 。 

请 注意 ,数组 问题 与 字符 串 问 题 往 往 是 相通 的 。 换 名 话说 ， 书 中 提 到 的 数组 问题 也 可 能 以 字 
符 串 的 形式 出 现 ， 反 之 亦 然 。 

1. 散 列表 

散 列 表 是 一 种 将 键 ( key ) 映射 为 值 (value ) 从 而 实现 快速 查找 的 数据 结构 。 在 简易 实现 中 ， 
散 列 表 包 含 一 个 底层 数组 和 一 个 散 列 函数 (hash function )。 插入 一 个 对 象 及 对 应 的 键 时 ， 散 列 函 
数 会 将 键 映 射 为 数组 的 一 个 索引 。 然 后 ， 这 个 对 象 就 会 储存 到 数组 中 该 索引 对 应 的 位 置 。 

不 过 , 通常 情况 下 ， 这 个 方法 还 不 够 完善 。 在 上 面 的 实现 中 , 所 有 可 能 的 键 必须 转化 为 各 不 
相同 的 散 列 值 ， 否 则 一 不 小 心 就 可 能 改写 某 些 数据 。 因 此 ， 为 了 防止 这 类 “碰撞 冲突 ”， 这 个 数 
组 会 变 得 非常 大 ， 以 便 放 下 所 有 可 能 的 键 。 

除了 创建 按 索引 hash(key) 储 存 对 象 的 超大 数组 ， 我 们 还 可 以 选用 小 得 多 的 数组 ， 并 将 对 象 
储存 在 索引 为 hash(key) % array_length 的 数组 元 素 指向 的 链表 中 。 要 通过 某 个 键 来 查找 对 象 ， 
就 必须 根据 散 列 值 找到 对 应 的 链表 ， 然 后 在 链表 中 查找 相应 的 键 。 

另外 , 我 们 还 可 以 采用 二 义 查 找 树 来 实现 散 列 表 。 只 要 我 们 让 这 棵 树 保 持平 衡 ， 就 能 保证 数 
据 查 找 用 时 为 O(log n)。 此 外 ， 这 种 实现 占用 的 空间 可 能 更 少 ， 原 因 很 简单 ， 我 们 不 必 一 开始 就 
分 配 一 个 大 数组 。 

面试 之 前 ， 建议 你 多 加 练习 ,掌握 散 列 表 的 实现 和 用 法 。 散 列表 是 面试 中 最 常见 的 数据 结构 
之 一 ， 相 关 问 题 也 是 技术 面试 的 常客 。 

下 面 是 一 段 使 用 散 列表 的 简单 Java 程 序 。 





















































1 public HashMap<Integer, Student> buildMap(Student[] students) { 

和 2 HashMap<Integer, Student> map = new HashMap<Integer, Student>(); 
3 for (Student s : students) map.put(s.getId(), s); 

4 return map; 

5 


} 
注意 ,尽管 有 时 面试 官 会 明确 要 求 使 用 散 列 表 , 但 多 半 还 是 要 靠 你 自己 想到 用 散 列 表 解 决 问题 。 
2. ArrayList (动态 数组 ) 
ArrayList， 即 动态 数组 ， 是 一 种 按 需 动态 调整 大 小 的 数组 ， 数 据 访 问 时 间 为 0(1)。 一 种 典型 
的 实现 是 在 数组 存 满 时 将 其 扩容 两 倍 。 每 次 扩容 用 时 O(n)， 不 过 这 种 操作 频次 极 少 ， 因 此 均 挫 下 
来 访问 时 间 仍 为 0(1)。 








1 public ArrayList<String> merge(String[] words, String[] more) { 
2 ArrayList<String> sentence = new ArrayList<String>(); 

3 for (String w : words) sentence.add(w); 

4 for (String w : more) sentence.add(w); 

5 return sentence; 

6 


} 
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所 有 字符 申 等 长 ( 皆 为 r)， 一 共有 /个 字符 申 。 


符 


3. StringBuffer 
假设 你 要 将 一 组 字符 串 拼 接 起 来 , 如 下 所 示 。 这 段 代 码 会 运行 多 长 时 间 ? 为 简单 起 见 , 假设 














1 public String joinWords(String[] words) { 

2 String sentence = “人 2; 

3 for (String w : words) { 

4 sentence = Sentence + WwW; 

5 } 

6 return sentence; 

7 } 

每 次 拼接 都 会 新 建 一 个 字符 串 ， 包 含 原 有 两 个 字符 串 的 全 部 字符 。 第 一 次 循环 要 拷贝 x 个 字 
第 二 次 循环 要 拷贝 2x 个 字符 , 第 三 次 要 拷 3x 个 , 依 此 类 推 。 综 上 , 这 段 代码 的 时 间 开 销 为 Ox 



































+2x 十 … 十 nx)， 可 简化 为 OGxn”))。 为 什么 不 是 OGxn")?” 因 为 1 +2+… 十 n 等 于 n(n+1)/2， 即 O(n”)。 


StringBuffer 可 以 避免 上 面 的 问题 。 它 会 直接 创建 一 个 足以 容纳 所 有 字符 串 的 数组 ， 等 到 





拼接 完成 才 将 这 些 字符 串 转 成 一 个 字符 申 。 





1 public String joinWords(String[] words) { 

2 StringBuffer sentence = new StringBuffer(); 
3 for (String w : words) { 

4 sentence.append(w); 

5 } 

6 return sentence.toString(); 

7 


} 
不 妨 试 着 自己 实现 一 把 StringBuffer， 这 对 你 掌握 字符 串 、 数 组 和 常见 数据 结构 大 有 神 益 。 





面试 题目 





1.1 


1.2 
1.3 


1.4 


1.5 








实现 一 个 算法 ,确定 一 个 字符 串 的 所 有 字符 是 否 全 都 不 同 ,假使 不 允许 使 用 额外 的 数据 结构 ， 
又 该 如 何 处 理 ? (第 108 页 ) 

用 C 或 C++ 实现 void reverse(char* str) 国 数 ， 即 反 转 一 个 null 结 尾 的 字符 串 。( 第 109 页 ) 
给 定 两 个 字符 串 , 请 编写 程序 , 确定 其 中 一 个 字符 串 的 字符 重新 排列 后 ,能 否 变 成 男 一 个 字 
符 串 。( 第 109 页 ) 

编写 一 个 方法 ， 将 字符 串 中 的 空格 全 部 替换 为 “%20”。 假 定 该 字符 串 尾部 有 足够 的 空间 存 
放 新 增 字符 ， 并 且 知 道 字符 串 的 “真实 ”长 度 。( 注 : 用 Java 实 现 的 话 ， 请 使 用 字符 数组 实 
现 ， 以 便 直接 在 数组 上 操作 。) (第 111 页 ) 

示例 

输入 : “Mr John Smith ” 

输出 : “Mr%26John%26Ssmith” 

利用 字符 重复 出 现 的 次 数 ， 编 写 一 个 方法 ， 实 现 基 本 的 字符 串 压 缩 功能 。 比 如 ， 字符 
串 “aabcccccaaa” 会 变 为 “a2blc$a3”。 若 “压缩 ”后 的 字符 串 没 有 变 短 ， 则 返回 原先 的 字 
符 串 。( 第 112 页 ) 
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1.6 给 定 一 幅 由 NxN 和 矩阵 表示 的 图 像 ， 其 中 每 个 像素 的 大 小 为 4 字 节 ， 编 写 一 个 方法 ， 将 图 像 旋 
转 90 度 。 不 占用 额外 内 存 空 间 能 否 做 到 ? (第 114 页 ) 

1.7 编写 一 个 算法 ， 若 MxN 算 阵 中 某 个 元 素 为 0， 则 将 其 所 在 的 行 与 列 清 零 。( 第 115 页 ) 

1.8 假定 有 一 个 方法 issubstring， 可 检查 一 个 单词 是 否 为 其 他 字符 串 的 子 串 。 给 定 两 个 字符 串 
s1 和 s2， 请 编写 代码 检查 s2 是 否 为 s1 旋 转 而 成 ， 要 求 只 能 调用 一 次 isSsubstring。( 比如 ， 
waterbottle 是 erbottlewat 旋 转 后 的 字符 串 。) (第 116 页 ) 



































参考 问题 : 位 操作 (#5.7 ) ; 面向 对 象 设计 (#8.10 ) ; 递归 (#9.3 ) ; 排序 与 查找 (#11.6 ) ; 
C++ (#13.10 ) ; 中 等 难题 (#17.7、#17.8、#17.14 ) 。 


8.2 ”链表 


链表 问题 有 时 会 难 倒 不 少 求职 者 ， 因 为 链表 元 素 访 问 用 时 不 定 ， 而 且 往往 涉及 递归 。 不 过 ， 
好 在 链表 问题 不 是 变化 多 端 ， 许 多 问题 只 是 在 常见 问题 的 基础 上 稍 作 调 整 而 已 。 

链表 问题 非常 依赖 基本 概念 ， 对 于 求职 者 来 说 , 可 以 从 无 到 有 实现 链表 是 一 项 基本 要 求 。 下 
面 是 我 们 给 出 的 参考 实现 代码 。 

1. 创建 链表 

下 面 的 代码 实现 了 一 个 非常 基本 的 单 向 链表 。 




















1 class Node { 

2 Node next = null; 

3 int data; 

4 

5 public Node(int d) { 

6 data = d; 

7 } 

8 

9 void appendToTail(int d) { 
16 Node end = new Node(d); 
11 Node n = this; 

12 while (n.next != null) { 
13 n = n.next; 

14 } 

工 5 n.next = end 

16 } 

17 } 








切记 ， 在 面试 中 遇 到 链表 题 时 ， 务 必 弄 清楚 它 到 底 是 单 向 链表 还 是 双向 链表 。 

2. 删除 单 向 链表 中 的 结 点 

删除 单 向 链表 中 的 结 点 非常 简单 。 给 定 一 个 结 点 n， 我 们 先 找到 它 的 前 趋 结 点 prev， 并 将 
prev.next 设 置 为 n.next。 如 果 这 是 双向 链表 , 我 们 还 要 更 新 n.next , 将 n.next.prev 置 为 n.prev。 
当然 ， 我 们 必须 注意 : (1) 检查 空 指针 ; (2) 必要 时 更 新 表 头 〈head ) 或 表 尾 tail ) 指针 。 

此 外 ， 如 果 采 用 C、C++ 或 其 他 要 求 开 发 人 员 自 行 管 理 内 存 的 语言 ， 还 应 考虑 要 不 要 释放 删 
除 结 点 的 内 存 。 
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1 Node deleteNode(Node head, int d) { 
2 Node n = head; 

3 

4 if (n.data == d) { 

5 return head.next; /* 表 头 指向 下 一 结 点 */ 
6 } 

7 

8 while (n.next != null) { 

9 if (n.next.data == d) { 

16 n.next = n.next.next; 

11 return head; /* 表 头 不 变 */ 
12 } 

13 n = n.next; 

14 } 

15 return head; 

16 } 


3. “ 快 行 指针 ”技巧 

在 处 理 链表 问题 时 ,“ 快 行 指 针 ”( runner, 或 称 第 二 个 指针 ) 是 一 种 很 常见 的 技巧 。“ 快 行 指 
针 ” 指 的 是 同时 用 两 个 指针 来 迭代 访问 链表 ， 只 不 过 其 中 一 个 比 男 一 个 超前 一 些 。“ 快 ”指针 往 
往 先行 几 步 ， 或 与 “ 慢 ” 指 针 相 差 固定 的 步 数 。 

举 个 例子 ， 假 定 有 一 个 链表 al->az->…->an->bl->b->…->bn， 你 想 将 其 重新 排列 成 
ai->bli->az->b->…->an->by 。 另 外 ， 你 不 知道 该 链表 的 长 度 〈 但 确定 它 有 偶数 个 元 素 )。 

你 可 以 用 两 个 指针 ， 其 中 p1( 快 指针 ) 每 次 都 向 前 移动 两 步 ， 而 同时 p2 只 移动 一 步 。 当 p1 
到 达 链 表 未 尾 时 ，p2 刚 好 位 于 链表 中 间 位 置 。 然 后 ， 再 让 p1 与 p2 一 步 步 从 尾 向 头 反 向 移动 , 并 将 
p2 指 向 的 结 点 搬入 到 p1 所 指 结 点 后 面 。 

4. 递归 问题 

许多 链表 问题 都 要 用 到 递归 。 解决 链 表 问 题 碰壁 时 ,不 妨 试 试 递归 法 能 否 奏效 。 这 里 暂时 不 
会 深入 探讨 递归 ， 后 面 会 有 专门 章节 予以 讲解 。 

当然 ,还 需 注意 递归 算法 至 少 要 占用 O(n) 空 间 ， 其 中 nn 为 递归 调用 的 层 数 。 实 际 上 ， 所 有 递 
归 算 法 都 可 以 转换 成 迭代 法 ， 只 是 后 者 实现 起 来 可 能 要 复杂 得 多 。 
































面试 题目 
2.1 编写 代码 ， 移 除 未 排序 链表 中 的 重复 结 点 。( 第 117 页 ) 
进 阶 
如 果 不 得 使 用 临时 缓冲 区 ， 该 怎么 解决 ? 
2.2 实现 一 个 算法 ， 找 出 单 向 链表 中 倒数 第 个 结 点 。( 第 118 页 ) 
2.3 实现 一 个 算法 ， 删 除 单 向 链表 中 间 的 某 个 结 点 ， 假 定 你 只 能 访问 该 结 点 。( 第 120 页 ) 
示例 


输入 : 单 向 链表 a->b->c->d->e 中 的 结 点 c。 
结果 : 不 返回 任何 数据 ， 但 该 链表 变 为 : a->b->d->e 
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2.4 编写 代码 ， 以 给 定 值 x 为 基准 将 链表 分 割 成 两 部 分 ， 所 有 小 于 x 的 结 点 排 在 大 于 或 等 于 x 的 结 
点 之 前 。( 第 121 页 ) 

2.5 给 定 两 个 用 链表 表示 的 整数 ， 每 个 结 点 包含 一 个 数位 。 这 些 数位 是 反 向 存放 的 ,也 就 是 个 位 
排 在 链表 首部 。 编 写 函 数 对 这 两 个 整数 求 和 ， 并 用 链表 形式 返回 结果 。( 第 123 页 ) 
示例 
输入 : (7-> 1 -> 6) + (5 -> 9 -> 2), 即 617 + 295。 
输出 : 2 -> 1 -> 9， 即 912。 




















进 阶 
假设 这 些 数位 是 正 向 存放 的 ， 请 再 做 一 遍 。 
示例 


输入 : (6 -> 1 -> 7) + (2 -> 9 -> 5), 即 617 + 295。 
输出 : 9 -> 1 -> 2， 即 912。 
2.6 给 定 一 个 有 环 链表 ， 实 现 一 个 算法 返回 环 路 的 开头 结 点 。( 第 126 页 ) 
有 环 链表 的 定义 
在 链表 中 某 个 结 点 的 next 元 素 指向 在 它 前 面 出 现 过 的 结 点 ， 则 表明 该 链表 存在 环 路 。 
示例 
输入 : A ->B ->C->D -> E ->C(C 结 点 出 现 了 两 次 )。 
输 ! i C 
2.7 编写 一 个 函数 ， 检 查 链 表 是 否 为 回 文 。( 第 128 页 ) 
参考 问题 : 树 与 图 (类 .4) ; 面向 对 象 设计 (#8.10 ) ; 扩展 性 与 内 存 限制 (#10.7) ; 中 等 
难题 (#17.13 ) 。 






































8.3” 栈 与 队列 


和 链表 问题 一 样 , 熟练 掌握 数据 结构 的 基本 原理 , 栈 与 队列 问题 处 理 起 来 要 容易 得 多 。 当然 ， 
有 些 问 题 也 可 能 相当 环 手 。 部 分 问题 不 过 是 对 基本 数据 结构 略 作 调整 ， 而 其 他 问题 则 要 难得 多 。 

1. 实现 一 个 栈 

栈 采 用 后 进 先 出 ( LIFO ) 顺序 。 换 言 之 ， 像 一 堆 盘 子 那样 ， 最 后 人 栈 的 元 素 最 先 出 栈 。 

下 面 给 出 了 栈 的 简单 实现 代码 。 注 意 ,， 栈 也 可 以 用 链表 实现 。 实 际 上 ， 栈 和 链表 本 质 上 是 一 
样 的 ， 只 不 过 用 户 通 带 只 能 看 到 栈 顶 元 素 。 




















class Stack { 
Node top; 


if (top != null) { 
Object item = top.data; 


于 
2 
3 
4 Object pop() { 
5 
6 
7 top = top.next; 
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8 return item; 

9 } 

16 return null; 

11 } 

12 

13 void push(Object item) { 
14 Node 七 = new Node(item); 
15 t.next = top; 

16 top = 七 ; 

7 } 

18 

19 Object peek() { 

26 return top.data; 

21 } 

22 } 


2. 实现 一 个 队列 

队列 采用 先进 先 出 (FIFO ) 顺序 。 就 像 一 支 排队 购 票 的 队伍 那样 ,最 早 入 列 的 元 素 也 是 最 先 
出 列 的 。 

队列 也 可 以 用 链表 实现 ， 新 增 元 素 追 加 至 表 尾 。 


1 class Queue { 











2 Node first, last; 

E: 

4 void enqueue(Object item) { 
5 if (first == null) { 

6 last = new Node(item); 
7 first = last; 

8 } else { 

9 last.next = new Node(item); 
16 last = last.next; 

盾 } 

12 } 

13 

14 Object dequeue() { 

15 if (first != null) { 

16 Object item = first.data; 
17 first = first.next; 

18 return item; 

19 } 

26 Peturn null; 

2 } 

22 } 

面试 题目 





3.1 描述 如 何 只 用 一 个 数组 来 实现 三 个 栈 。( 第 131 页 ) 
3.2 请 设计 一 个 栈 ， 除 pop 与 push 方 法 ,还 支持 min 方 法 ,可 返回 栈 元 素 中 的 最 小 值 。push、pop 
和 min 三 个 方法 的 时 间 复 杂 度 必须 为 O(1)。( 第 135 页 ) 
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3.3 设想 有 一 堆 盘 子 , 堆 太 高 可 能 会 倒 下 来 。 因 此 ， 在 现实 生活 中 ,盘子 堆 到 一 定 高 度 时 ， 我们 
就 会 男 外 堆 一 堆 盘 子 。 请 实现 数据 结构 setofstacks， 模 拟 这 种 行为 。setofstacks 应 该 由 
多 个 栈 组 成 ， 并 且 在 前 一 个 栈 填 满 时 新 建 一 个 栈 。 此 外 ，Ssetofstacks.push() 和 
Setofstacks.pop() 应 该 与 普通 栈 的 操作 方法 相同 (也 就 是 说 ，pop() 返 回 的 值 ， 应 该 跟 只 
有 一 个 栈 时 的 情况 一 样 )。( 第 137 页 ) 

进 阶 
实现 一 个 popAt (int index) 方 法 ， 根 据 指定 的 子 栈 ， 执 行 pop 操 作 。 

3.4 在 经 典 问题 汉 庄 塔 中 ， 有 3 根 柱子 及 N 个 不 同 大 小 的 穿孔 圆 盘 ， 盘 子 可 以 滑 和 人 任意 一 根 柱子 。 
一 开始 , 所 有 盘子 自 底 向 上 从 大 到 小 依次 套 在 第 一 根 柱子 上 ( 即 每 一 个 盘子 只 能 放 在 更 大 的 
盘子 上 面 )。 移动 圆 盘 时 有 以 下 限制 .; 

(1) 每 次 只 能 移动 一 个 盘子 ; 

(2) 盘子 只 能 从 柱子 顶端 滑 出 移 到 下 一 根 柱子 ; 

(3) 盘子 只 能 县 在 比 它 大 的 盘子 上 。 

请 运用 栈 ， 编 写 程序 将 所 有 盘子 从 第 一 根 柱子 移 到 最 后 一 根 柱子 。( 第 140 页 ) 

3.5 实现 一 个 MyQueue 类 ， 该 类 用 两 个 栈 来 实现 一 个 队列 。( 第 142 页 ) 

3.6 编写 程序 ， 按 升序 对 栈 进行 排序 ( 即 最 大 元 素 位 于 栈 顶 )。 最 多 只 能 使 用 一 个 额外 的 栈 存放 

临时 数据 , 但 不 得 将 元 素 复 制 到 别 的 数据 结构 中 ( 如 数组 ), 该 栈 支 持 如 下 操作 : push、pop、 

peek 和 isEmpty。( 第 143 页 ) 

有 家 动物 收容 所 只 收容 狗 与 猫 , 且 严 格 遵守 “先进 先 出 ”的 原则 。 在 收养 该 收容 所 的 动物 时 ， 

收养 人 只 能 收养 所 有 动物 中 “最 老 ”( 根据 进入 收容 所 的 时 间 长 短 ) 的 动物 , 或者， 可 以 挑 

选 猫 或 狗 ( 同时 必须 收养 此 类 动物 中 “最 老 ” 的 )。 换 言 之 ， 收 养 人 不 能 自由 挑选 想 收养 的 

对 象 。 请 创建 适用 于 这 个 系统 的 数据 结构 , 实现 各 种 操作 方法 , 比如 enqueue 、dequeueAny、 

dequeueDog 和 dequeueCat 等 。 人 允许 使 用 Java 内 置 的 LinkedList 数 据 结构 。( 第 145 页 ) 


参考 问题 : 链表 ( 需 .7) ; 数学 与 概率 (#7.7) 。 


8.4 树 与 图 


许多 求职 者 会 觉得 树 与 图 的 问题 是 最 难 对 付 的 。 检索 这 两 种 数据 结构 比 数组 或 链表 等 线性 数 
据 结 构 要 复杂 得 多 。 此 外 ， 在 最 坏 情况 和 平均 情况 下 ， 检 索 用 时 可 能 差别 巨大 ， 对 于 任意 算法 ， 
我 们 都 要 从 这 两 方面 进行 评估 。 能 够 自如 地 从 无 到 有 实现 树 或 图 对 求职 者 而 言 非常 重要 。 

. 需要 注意 的 潜在 问题 

树 与 图 的 问题 容易 出 现 含糊 的 细节 和 错误 的 假设 。 务 必 留 意 下 列 问 题 ， 必 要 时 寻求 淤 清 。 

@ 二 又 树 与 二 又 查找 树 

碰 到 二 义 树 问 题 时 , 许多 求职 者 会 假定 面试 官 问 的 是 二 又 查找 树 。 此 时 务必 问 清 楚 二 义 树 是 
和 否 为 二 又 查找 树 。 二 又 查找 树 附 加 有 如 下 条 件 : 对 于 任意 结 点 ， 左 子 结 点 小 于 或 等 于 当前 结 点 ， 
后 者 又 小 于 所 有 右 子 结 点 。 
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@ 平衡 与 不 平衡 

许多 树 都 是 平衡 的 , 但 并 非 全 都 如 此 。 树 是 否 平衡 要 找 面 试 官 确认 。 如 果树 是 不 平衡 的 ,你 
应 当 从 平均 情况 和 最 坏 情 况 所 需 时 间 来 描述 自己 的 算法 。 注 意 , 树 的 平衡 有 多 种 方法 , 平衡 一 棵 
树 只 意味 着 子 树 的 深度 差 不 会 超过 一 定 值 ， 并 不 表示 左 子 树 和 右 子 树 的 深度 完全 相同 。 

@ 完满 和 完整 《Full and Complete ) 

完满 和 完整 树 的 所 有 叶 结 点 都 在 树 的 底部 , 所 有 非 叶 结 点 都 有 两 个 子 结 点 。 注 意 完满 和 完整 
树 极其 稀少 ， 因 为 一 棵 树 必须 正好 有 2” - 1 个 结 点 才能 满足 这 个 条 件 。 



































2. 二 叉 树 遍历 
面试 之 前 ， 你 应 该 能 够 熟练 实现 中 序 、 后 序 和 前 序 遍 历 。 其 中 最 常见 的 是 中 序 遍 历 ， 先 遍历 


左 子 树 ， 然 后 访问 当前 结 点， 最 后 遍历 右 子 树 。 

3. 树 的 平衡 ， 红 黑 树 和 平衡 二 又 树 

学 习 如 何 实现 平衡 树 可 助 你 成 为 更 好 的 软件 工程 师 ， 只 不 过 面试 中 很 少 会 问 及 平衡 树 。 你 应 
该 熟悉 平衡 树 各 种 操作 的 执行 时 间 ， 大致 了 解 如 何平 衡 一 棵 树 。 当 然 ， 就 面试 而 言 ， 掌 握 个 中 细 
节 也 许 没什么 必要 。 

4. 单词 查找 树 (trie) 

trie 树 是 n 层 树 的 一 种 变 体 ， 其 中 每 个 结 点 存储 有 字符 。 整 棵 树 的 每 条 路 径 自 上 而 下 表示 一 个 
单词 。 一 棵 简单 的 trie 树 类 似 下 图 : 











5. 图 的 遍历 

大 部 分 求职 者 都 比较 熟悉 二 又 树 的 遍历 ， 但 图 的 遍历 则 要 难得 多 。 广 度 优先 搜索 ( BFS ) 更 
是 难 上 加 难 。 

值得 注意 的 是 ,广度 优先 搜索 ( BFS ) 和 深度 优先 搜索 ( DFS ) 通常 用 于 不 同 的 场景 。 如 要 访 
问 图 中 所 有 结 点 "， 或 者 访问 最 少 的 结 点 直至 找到 想 找 的 结 点 ，DEFS 一 般 最 为 简单 。 不 过 ， 如 果 一 
棵 树 的 规模 非常 大 ， 在 离 最 初 结 点 太 远 时 想 要 随时 退出 的 话 ，DFS 可 能 会 有 问题 ; 我们 可 能 搜索 
了 该 结 点 的 成 千 上 万 个 祖先 结 点 , 却 还 未 搜索 该 结 点 的 全 部 子 结 点 。 对 于 这 些 情况 , 一般 首 选 BFS。 





























Qz 图 中 的 “ 结 点 ”一 般 称 为 顶点 ， 这 里 依 原 文 译作 结 点 。 一 一 译 者 注 
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@ 深度 优先 搜索 (DFS ) 

在 DFS 中 ,我 们 会 访问 结 点 r， 然 后 循环 访问 r 的 每 个 相 邻 结 点 。 在 访问 r 的 相 邻 结 点 n 时 ,我 
们 会 在 继续 访问 r 的 其 他 相 邻 结 点 之 前 先 访问 n 的 所 有 相 邻 结 点 。 也 就 是 说 ， 在 继续 搜索 r 的 其 他 
子 结 点 之 前 ， 我 们 会 先 穷 尽 搜 索 n 的 子 结 点 。 

注意 ， 前 序 和 树 遍 历 的 其 他 形式 都 是 一 种 DFS。 主 要 区 别 在 于 ， 对 图 实现 该 算法 时 ， 我 们 必 
须 先 检查 该 结 点 是 否 已 访问 。 如 果 不 这 么 做 ， 就 可 能 陷入 无 限 循 环 。 

下 面 是 实现 DFS 的 伪 代 码 。 





























1 void search(Node root) { 

2 if (root == null) return; 

3 visit(root); 

4 root.visited = true; 

5 foreach (Node n in root.adjacent) { 
6 if (n.visited == false) { 

7 search(n); 

8 } 

9 } 

10 } 


@ 广度 优先 搜索 ( BFS ) 

BFS 相 对 不 太 直 观 ， 除 非 之 前 熟悉 BFS 的 实现 ， 否 则 大 部 分 求职 者 都 会 觉得 它 很 难 对 付 。 

在 BFS 中 ,我 们 会 在 搜索 r 的 “孙子 结 点 ”之 前 先 访问 结 点 r 的 所 有 相 邻 结 点 。 用 队列 实现 的 
迭代 方案 往往 最 有 效 。 

1 void search(Node root) { 

2 Queue queue = new Queue(); 





3 root.visited = true; 

4 visit(root); 

5 queue.enqueue(root); // 加 至 队列 尾部 
6 

7 while (!queue.isEmpty()) { 

8 Node r = queue.dequeue(); // 从 队列 头 部 移 除 
9 foreach (Node n in r.adjacent) { 
16 if (n.visited == false) { 

11 visit(n); 

12 n.visited = true; 

13 queue.enqueue(n); 

14 } 

15 } 

16 } 

17 } 





当面 试 官 要 求 你 实现 BFS 时 ， 切 记 关键 在 于 队列 的 使 用 。 用 了 队列 ， 这 个 算法 的 其 余部 分 自 
然 也 就 成 型 了 。 








面试 题目 





4.1 实现 一 个 函数 ， 检 查 二 又 树 是 否 平衡 。 在 这 个 问题 中 ， 平 衡 树 的 定义 如 下 : 任意 一 个 结 点 ， 
其 两 棵 子 树 的 高 度 差 不 超 过 1。( 第 146 页 ) 
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4.2 给 定 有 向 图 ， 设 计 一 个 算法 ， 找 出 两 个 结 点 之 间 是 否 存 在 一 条 路 径 。( 第 148 页 ) 

4.3 给 定 一 个 有 序 整 数 数组 ,元 素 各 不 相同 且 按 升序 排列 , 编写 一 个 算法 , 创建 一 棵 高 度 最 小 的 
二 又 查 找 树 。( 第 149 页 ) 

4.4 给 定 一 棵 二 叉 树 ,设计 一 个 算法 ,创建 含有 某 一 深度 上 所 有 结 点 的 链表 ( 比如 ,， 若 一 棵 树 的 
深度 为 D， 则 会 创建 出 D 个 链表 )。( 第 150 页 ) 

4.5 实现 一 个 函数 ， 检 查 一 棵 二 叉 树 是 否 为 二 又 查 找 树 。( 第 151 页 ) 

4.6 设计 一 个 算法 ， 找 出 二 又 查找 树 中 指定 结 点 的 “下 一 个 ” 结 点 (也 即 中 序 后 继 )。 可 以 假定 
每 个 结 点 都 含有 指向 父 结 点 的 连接 。( 第 154 页 ) 

4.7 设计 并 实现 一 个 算法 , 找 出 二 又 树 中 某 两 个 结 点 的 第 一 个 共同 祖先 。 不 得 将 额外 的 结 点 储存 
在 另外 的 数据 结构 中 。 注 意 : 这 不 一 定 是 二 又 查找 树 。( 第 155 页 ) 





























4.8 你 有 两 棵 非常 大 的 二 叉 树 : T1， 有 几 百 万 个 结 点 ; IT2， 有 几 百 个 结 点 。 设 计 一 个 算法 ， 判 
汤 T2 是 否 为 T1 的 子 树 。 


如 果 T1 有 这 么 一 个 结 点 n， 其 子 树 与 T2 一 模 一 样 ， 则 T2 为 TI 的 子 树 。 也 就 是 说 ， 从 结 点 n 处 
把 树 砍 断 ， 得 到 的 树 与 T2 完 全 相同 。( 第 159 页 ) 

4.9 给 定 一 棵 二 又 树 ， 其 中 每 个 结 点 都 含有 一 个 数值 。 设 计 一 个 算法 ， 打 印 结 点 数值 总 和 等 于 某 个 
给 定 值 的 所 有 路 径 。 注 意 , 路 径 不 一 定 非得 从 二 又 树 的 根 结 点 或 叶 结 点 开始 或 结束 。( 第 161 页 ) 


参考 问题 : 扩展 性 与 存储 限制 (#10.2、#10.5 ) ; 排序 与 查找 (#11.8 ) ; 中 等 难题 (#17.13、 
#17.14 ) ; 高 难度 题 (#18.6、#18.8、#18.9、#18.10、#18.13 ) 。 


8.5 位 操作 


位 操作 可 以 用 于 解决 各 种 各 样 的 问题 。 有 时 候 ， 有 的 问题 会 明确 要 求 用 位 操作 来 解决 ， 而 在 
其 他 情况 下 ,位 操作 也 是 优化 代码 的 实用 技巧 。 写 代码 要 熟悉 位 操作 ,同时 也 要 熟练 掌握 位 操作 
的 手工 运算 。 处 理 位 操作 问题 时 ,务必 小 心 辟 到, 不经意 间 就 会 犯 下 各 种 小 错 。 代 码 写 好 后 一 定 
要 进行 充分 的 测试 ， 也 可 以 边 写 代码 边 测试 。 

1. 手工 位 操作 

如 果 像 很 多 人 一 样 ,你 也 惯 怕 位 操作 问题 ,以 下 这 些 练习 对 你 大 有 神 益 。 当 你 一 筹 莫 展 或 办 
惑 不 解 时 ， 不 妨 换 用 十 进 制 来 理解 相关 操作 ， 再 将 这 些 操作 过 程 应 用 到 二 进 制 上 。 

记 住 , 符号 ^ 表 示 XOR ( 异 或 ) 操作 ,~ 表示 非 ( 取 反 ) 操作 。 为 简单 起 见 ， 假 定 操作 数 的 位 






















































































宽 为 4 位 。 我 们 可 以 手工 或 是 施 以 奉 干 技巧 (详情 如 下 ) 解决 下 表 第 三 列 的 问题 。 


9116 + 9616 8611 * 6161 9116 + 9116 
96611 + 9616 8611 * 6611 8166 * 6611 











69116 - 688611 1161 >> 2 1161 ^ (~1161) 
1666 - 6116 1161 ^ 6161 1611 & (~6 << 2) 














答案 : 第 一 行 (16686，1111，1166 ) ; 第 二 行 (6161，1661，1166 ) ; 第 三 行 (6611，6611，1111 ) ; 第 四 行 
(66196，1688，1666 ) 。 
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第 三 列 问题 的 解决 技巧 如 下 。 

(1) 68116 + 6116 相 当 于 6116 * 2， 也 就 是 将 6116 左 移 1 位 。 

(2) 9166 等 于 4，6166 * 6611 也 就 是 将 6611 乘 以 4。 一 个 数 与 2" 相 乘 ， 相 当 于 将 这 个 数 左 移 n 
位 。 于 是 ， 将 8e11 左 移 2 位 得 到 116e6。 

(3) 逐个 比特 分 解 这 一 操作 。 一 个 比特 与 对 它 取 反 的 值 做 异 或 操作 , 结果 总 是 1。 因此 , a^(~a) 
的 结果 是 一 串 1。 

(4) 类 似 x & (~8 “< n) 的 操作 会 将 x 最 右边 的 n 位 清 零 。~e 的 值 就 是 一 串 1， 将 它 左 移 n 位 后 
的 结果 为 一 串 1 后 面 跟 a 个 0。 将 这 个 数 与 x 进行 “位 与 ”操作 ， 相 当 于 将 x 最 右边 的 n 位 清 零 。 
要 处 理 其 他 问题 ,不 妨 打 开 Windows 上 的 计算 器 ， 选 择 “ 查 看 ”( View ) 菜单 项 ， 再 点 选 该 
工具 的 “程序 员 ”( Programmer ) 版 本 。 有 了 这 个 应 用 程序 ， 就 可 以 执行 位 与 、 异 或 和 移 位 等 各 
种 二 进 制 运算 。 

2. 位 操作 原理 与 技巧 

处 理 位 操作 问题 时 ,理解 以 下 原理 会 大 有 帮助 。 不 要 一 味 死记 硬 背 ， 而 应 思考 这 些 等 式 何以 
成 立 。 在 下 面 的 示例 中 ,“1s” 和 “0s” 分 别 表示 一 串 1 和 一 串 0。 
































XA^ 人 6@s = X Xx&6s=6 X | 6s = Xx 
X ^ 1s = ~x X & 1s = X x | 1s = 1s 
x^x=0 x&x=x X |X=xXx 


要 理解 这 些 表达 式 的 含义 , 你 必须 记 住 所 有 操作 是 按 位 进行 的 , 某 一 位 的 运算 结果 不 会 影响 
其 余 位 。 也 就 是 说 ， 只 要 上 述 语句 对 某 一 位 成 立 ， 则 同样 适用 于 一 串 位 。 

3. 常见 位 操作 : 获取 、 设 置 、 清 除 及 更 新 位 数据 

以 下 这 些 位 操作 非常 重要 ， 不 过 切忌 死记 硬 背 ， 否 则 只 会 滋生 一 些 难以 察觉 的 错误 。 相 反 ， 
你 要 吃透 这 些 操 作 方法 ， 学 会 举一反三 ， 灵 活 处 理 其 他 问题 。 

@ 获取 

该 方法 将 1 左 移 ; 位 ， 得 到 形 如 8ee16888 的 值 。 接 着 ， 对 这 个 值 与 num 执 行 “ 位 与 ”操作 ， 从 
而 将 位 之 外 的 所 有 位 清 零 。 最 后 ， 检 查 该 结果 是 否 为 零 。 不 为 零 说 明 i 位 为 1!， 否 则 ，i 位 为 0。 

1 boolean getBit(int num, int i) { 

2 return ((num & (1 << i)) != 0); 

3 } 

@ 置 位 

setBit 先 将 1 左 移 i 位 ， 得 到 形 如 eee1686e6 的 值 。 接 着 ， 对 这 个 值 和 num 执 行 “位 或 ”操作 ， 
这 样 只 会 改变 i 位 的 数据 。 该 扼 码 ;位 除外 的 位 均 为 零 ， 故 而 不 会 影响 num 的 其 余 位 。 

1 int setBit(int num, int i) { 

2 return num | (1 << i); 

3 } 

@ 清 堆 

该 方法 与 setBit 刚 好 相反 。 首 先 , 将 1 左 移 ;位 取得 形 如 8ee18eee 的 值 , 对 这 个 值 取 反 进而 得 
到 类 似 11161111 的 掩 码 。 接 着 ， 对 该 掩 码 和 num 执 行 “ 位 与 ”操作 。 这 样 只 会 清 零 num 的 位， 其 
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余 位 


接着 


则 保持 不 变 。 


1 int clearBit(int num, int i) { 
2 int mask = ~(1 << i); 

3 return num & mask; 
4 


} 
将 num 最 高 位 至 ;位 ( 含 ) 清 零 的 做 法 如 下 : 


1 int clearBitsMSBthroughI(int num, int i) { 


2 int mask = (1 << i) - 1; 
3 return num & mask; 
4 1} 


将 i 位 至 0 位 ( 含 ) 清 零 的 做 法 如 下 : 


1 int clearBitsIthroughe(int num, int i) { 
之 int mask = ~((1 << (i+1)) - 1); 
3 return num & mask; 
4 } 
更 新 





e@ 
这 个 方法 将 setBit 与 clearBit 合 二 为 一 。 首 先 ,， 用 诸如 11181111 的 手 码 将 num 的 第 ;位 清 零 。 
， 将 待 写 人 值 \ 左 移 ;i 位 ， 得 到 一 个 ;位 为 v 但 其 余 位 都 为 0 的 数 。 最 后 ， 对 之 前 取得 的 两 个 结 


执行 “位 或 ”操作 ，v 为 1 则 将 num 的 i 位 更 新 为 1， 否 则 该 位 仍 为 0。 





1 int updateBit(int num, int i, int v) { 
2 int mask = ~(1 << i); 

3 return (num & mask) | (v << i); 

4 


} 
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5.1 


5.2 


5.3 


5.4 
5.5 


给 定 两 个 32 位 的 整数 N 与 MY， 以 及 表示 比特 位 置 的 i 与 i。 编 写 一 个 方法 ,将 M 插 入 N， 使 得 
从 NN 的 第 j 位 开始 ， 到 第 ;位 结束 。 假定 从 7 位 到 i 位 足以 容纳 M， 也 即 知 M=10011， 那么 j 和 i 之 间 
至 少 可 容纳 5 个 位 。 例 如 ， 不 可 能 出 现 /=3 和 ;= 2 的 情况 ， 因 为 第 3 位 和 第 2 位 之 间 放 不 下 M。 
(第 163 页 ) 

示例 

输入 : N = 16666666666，M = 16611， i = 2, j=6 

输出 : N = 16661661160 

给 定 一 个 介 于 0 和 1 之 间 的 实数 (如 0.72 )， 类 型 为 double,， 打印 它 的 二 进 制 表示 。 如 果 该 数字 
无 法 精确 地 用 32 位 以 内 的 二 进 制 表示 ， 则 打印 “ERROR”。( 第 164 页 ) 

给 定 一 个 正 整数 ， 找 出 与 其 二 进 制 表 示 中 1 的 个 数 相 同 、 且 大 小 最 接近 的 那 两 个 数 (一 个 略 
大 , 一 个 略 小 )。( 第 165 页 ) 

解释 代码 ((n & (n-1)) == 8) 的 有 具体 含义 。( 第 170 页 ) 

编写 一 个 函数 ， 确 定 需 要 改变 几 个 位 ， 才 能 将 整数 4 转 成 整数 B。( 第 171 页 ) 

示例 
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输入 : 31，14 

输出 : 2 

5.6 编写 程序 , 交换 某 个 整数 的 奇数 位 和 偶数 位 , 使 用 指令 越 少 越 好 (也 就 是 说 , 位 0 与 位 1 交换 ， 
位 2 与 位 3 交换 ， 依 此 类 推 )。( 第 171 页 ) 

5.7 数组 A 包 含 0 到 ”的 所 有 整数 ， 但 其 中 缺 了 一 个 。 在 这 个 问题 中 ， 只 用 一 次 操作 无 法 取得 数组 
A 里 某 个 整数 的 完整 内 容 。 此 外 , 数组 A 的 元 素 丝 以 二 进 制 表 示 , 唯一 可 用 的 访问 操作 是 “从 
A[i] 取 出 第 7 位 数据 *"， 该 操作 的 时 间 复 杂 度 为 常数 。 请 编写 代码 找 出 那个 缺失 的 整数 。 你 有 
办 法 在 O(n) 时 间 内 完成 吗 ? (第 172 页 ) 

5.8 有 个 单 色 屏幕 存储 在 一 个 一 维 字 节 数组 中 ， 使 得 8 个 连续 像素 可 以 存放 在 一 个 字 节 里 。 屏 幕 
宽度 为 w， 且 w 可 被 8 整除 ( 即 一 个 字 节 不 会 分 布 在 两 行 上 )， 屏 幕 高 度 可 由 数组 长 度 及 屏幕 
宽度 推算 得 出 。 请 实现 一 个 函数 drawHorizontalLine(byte[] screen, int width，int x1， 
int x2，int y) ， 绘 制 从 点 Ce yy) 到 点 (x2, yy) 的 水 平 线 。( 第 174 页 ) 


参考 问题 : 数组 与 字符 串 (#1.1、#1.7) ; 递归 (#9.4、#9.11 ) ; 扩展 性 与 存储 限制 (#10.3、 
#10.4 ) ; C++ (#13.9 ) ; 中 等 难题 (#17.1、#17.4 ) ; 高 难度 题 (#18.1 ) 。 






































8.6 智力 题 


智力 题 当 属 最 有 争议 的 面试 题 之 列 , 很 多 公司 甚至 明文 规定 面试 中 不 得 出 现 智力 题 。 尽 管 如 
此 ， 你 还 是 会 时 不 时 地 磁 到 它 。 为 什么 呢 ? 因为 人 们 对 于 智力 题 尚 无 明确 的 定义 。 

不 过 ,好 在 哪怕 你 磁 到 了 这 类 问题 ， 一 般 来 说 它们 也 不 会 太 难 。 你 不 需要 做 脑筋 急 转 弯 ， 并 
且 几 乎 总 有 办 法 通过 逻辑 推理 得 出 答案 。 很 多 智力 题 甚 至 还 涉及 数学 或 计算 机 科学 的 基础 知识 。 

下 面 ， 我 们 会 列举 一 些 应 对 智力 题 的 常见 方法 。 

1. 大 声 说 出 你 的 思路 
遇 到 智力 题 时 , 切忌 惊 懂 。 就 像 算法 题 一 样 ， 面 试 官 只 不 过 想 看 看 你 会 如 何 处 理 难题 ; 他 们 
并 不 期 待 你 立即 给 出 正确 答案 。 只 管 大 声 说 出 解 题 思路 ， 让 面试 官 了 解 你 的 应 对 之 道 。 

2. 总 结 规律 和 模式 

很 多 情况 下 ， 你 会 发 现 ， 把 解 题 过 程 中 发 现 的 “规律 ”或 “模式 ” 写 下 来 帮助 很 大 。 并 且 ， 
你 确实 应 该 这 么 做 ， 这 有 助 于 加 深 记 忆 。 下 面 会 举例 说 明 这 种 方法 。 

给 定 两 条 绳子 , 每 条 强 子 燃烧 殖 尽 正好 要 用 一 个 小 时 。 怎样 用 这 两 条 绳子 准确 计量 15 分 钟 ? 
注意 这 些 绳子 密度 不 均匀 ， 因 此 烧 掉 半截 绳子 不 一 定 正好 用 半 个 小 时 。 

技巧 : 先 别 急 着 往 下 看 ， 不 妨 试 着 自己 解决 此 问题 。 一 定 要 看 下 面 的 提示 信息 的 话 
一 段 一 段 慢 慢 看 。 后 续 段 落 会 逐步 揭晓 答案 。 

从 题 日 可 知 , 计量 一 小 时 不 成 问题 。 当 然 也 可 以 计量 两 小 时 ， 先 点 燃 一 根 绳子 ， 等 它 燃 烧 殖 
尽 ， 再 点 燃 第 二 根 。 由 此 我 们 总 结 出 第 一 条 规律 。 

规律 1: 给 定 两 条 绳子 ， 燃 烧 殖 尽 各 需 x* 分 钟 和 ) 分 钟 ， 我 们 可 以 计时 x+y 分 钟 。 
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那么 , 还 有 其 他 人 烧 绳 子 的 花样 吗 ?” 当然 ,我 们 知道 从 中 间 ( 或 绳子 两 头 以 外 的 任意 位 置 ) 点 
燃 绳子 没什么 用 。 火 苗 会 向 绳子 两 头 草 延 ， 我 们 不 知道 过 多 和 久 才 会 烧 完 。 
话说 回来 ， 我 们 可 以 同时 点 燃 绳子 两 尖 。30 分 钟 后 火焰 便 会 在 绳子 某 个 位 置 汇 合 。 
规律 2: 给 定 一 条 需要 x 分 钟 烧 完 的 绳子 ， 我 们 可 以 计时 x/2 分 钟 。 
由 此 可 知 , 用 一 条 绳子 可 以 计时 30 分 钟 。 这 就 意味 着 我 们 可 以 在 燃烧 第 二 条 绳子 时 减 去 这 30 
分 钟 ， 也 就 是 点 燃 第 一 条 绳子 两 涉 的 同时 ， 只 点 燃 第 二 条 绳子 的 一 头 。 
规律 3: 烧 完 绳子 1 用 时 x 分 钟 , 绳子 2 用 时 y 分 钟 , 则 可 以 用 第 二 条 绳子 计时 (yx) 分 钟 或 (y-x/2) 
综合 以 上 规律 ， 不 难得 出 : 既然 可 以 用 绳子 2 计时 30 分 钟 ， 再 适时 点 燃 绳子 2 的 另 一 头 ( 见 规 
2 )， 则 1 分 钟 后 绳子 2 便 会 燃烧 殖 尽 。 
将 上 面 的 做 法 从 头 至 尾 整理 如 下 。 
(1) 点 燃 绳子 1 两 头 的 同时 ， 点 燃 绳子 2 的 一 头 。 
(2) 当 绳子 1 从 两 头 烧 至 中 间 某 个 位 置 时 ， 正 好 过 去 30 分 钟 。 而 绳子 2 还 可 以 再 烧 30 分 钟 。 
(3) 此 时 ， 点 燃 绳子 2 的 另 一 头 。 
(4) 15 分 钟 后 ， 绳 子 2 将 全 部 烧 完 。 
从 中 可 以 看 出 ， 只 要 一 步 步 归纳 规律 ， 并 在 此 基础 上 进行 总 结 ， 智 力 题 便 可 迎刃而解 。 
3. 略 作 变 通 
许多 智力 题 往 往 涉 及 将 最 坏 情况 减 至 最 低 限 度 的 问题 , 措辞 上 要 么 要 求 尽 可 能 减少 步骤 , 要 
么 限定 具体 的 试验 次 数 。 一 种 实用 的 技巧 是 尝试 “平衡 ”最 坏 情况 。 也 就 是 说 ， 如 果 早 先 的 解决 
方案 效果 不 太 理 想 ， 我 们 可 以 针对 最 坏 情况 略 作 变通 。 用 一 个 例子 来 解释 会 更 为 清晰 。 
“ 九 球 称 重 ”是 一 个 经 典 面试 题 。 给 定 9 个 球 ， 其 中 8 个 球 的 重量 相同 ， 只 有 一 个 比较 重 。 然 
后 给 定 一 个 天 平 ， 可 以 称 出 左右 两 边 哪 边 更 重 。 最 多 用 两 次 天 平 ， 找 出 这 个 重 球 。 
第 一 种 做 法 是 将 球 分 成 2 组 ，4 个 一 组 ， 第 9 个 球 暂 时 搁 在 一 边 。 如 果 有 一 组 球 较 重 ， 则 重 球 
必 在 其 中 ; 但 如 果 两 组 球 重量 相同 ， 则 第 9 个 球 为 重 球 。 按 此 思路 将 包含 重 球 的 这 一 组 球 再 分 成 
两 组 ， 在 最 坏 情况 下 我 们 需要 称 量 3 次 一 一 多 了 一 次 ! 
因此 ， 这 是 一 个 “失衡 ”的 解法 如果 第 9 个 球 是 重 球 ,我们 只 需 称 量 一 次 ; 但 如 果 不 是 ， 
则 需 称 量 3 次 。 如 果 我 们 略 作 调 整 , 将 更 多 的 球 与 第 9 个 球 配 在 一 起 , 就 不 会 出 现 “ 失 衡 ” 的 状况 。 
这 就 是 所 谓 “ 最 坏 情况 下 的 平衡 ”。 
现在 , 将 这 些 球 均 分 成 3 个 一 组 共 3 组 , 称 量 一 次 就 能 知道 哪 一 组 球 更 重 。 我们 甚至 可 以 总 结 
出 一 条 规律 : 给 定 N 个 球 ， 其 中 N 能 被 3 整除 ， 称 量 一 次 便 能 找到 包含 重 球 的 那 一 组 球 。 
找到 这 一 组 3 个 球 之 后 ， 我 们 只 是 简单 地 重复 此 前 的 模式 : 先 把 一 个 球 放 到 一 边 ， 称 量 剩 下 
的 两 个 球 。 从 中 挑 出 那个 重 球 ; 或 者 ， 如 果 这 两 个 球 重量 相同 ， 那 第 3 个 球 便 是 重 球 。 
4. 触 类 旁 通 
要 是 卡 膏 了, 不 妨 考虑 运用 前 面 提 到 的 算法 题 的 五 种 解法 。 剔 除 技术 层面 的 因素 ,智力 题 不 
外 平 就 是 些 算法 题 。 其 中 , 举例 法 、 简 化 推广 法 、 模 式 匹配 法 ,以 及 简单 构造 法 可 能 会 特别 有 用 。 
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面试 题目 
6.1 有 20 瓶 药 九 ， 其 中 19 瓶 装 有 1 克 / 粒 的 药丸 ， 余 下 一 瓶 法 有 1.1 克 / 粒 的 药丸 。 给 你 一 人 台 称 重 精 


准 的 天 平 ， 怎 么 找 出 比较 重 的 那 瓶 药丸 ?天 平 只 能 用 一 次 。( 第 175 页 ) 

6.2 有 个 8x8 棋 盘 ， 其 中 对 角 的 角落 上， 两 个 方 格 被 切 反 了 。 给 定 31 块 多 米 诺 上 骨牌， 一 块 骨 牌 恰 
好 可 以 覆盖 两 个 方 格 。 用 这 31 块 骨牌 能 否 盖 住 整个 棋盘 ? 请 证 明 你 的 答案 (提供 范例 , 或 证 
明 为 什么 不 可 能 )。( 第 176 页 ) 

6.3 有 两 个 水 过 ， 容 量 分 别 为 $ 夸 脱 (美制 : 1 夸 脱 =0.946 升 ， 英 制 : 1 夸 脱 =1.136 升 ) 和 3 和 夸 脱 ， 
若水 的 供应 不 限量 (但 没有 量 杯 )， 怎 么 用 这 两 个 水 壶 得 到 刚好 4 夸 脱 的 水 ? 注意 ,这 两 个 水 
壶 呈 不 规则 形状 ， 无 法 精准 地 装 满 “ 半 壶 ”水 。( 第 177 页 ) 

6.4 有 个 岛 上 住 着 一 和 群 人 ， 有 一 天 来 了 个 游客 , 定 了 一 条 奇怪 的 规矩 : 所 有 蓝 有 眼睛 的 人 都 必须 尽 
快 离开 这 个 岛 。 每 晚 8 点 会 有 一 个 航班 离岛 。 每 个 人 都 看 得 见 别 人 眼睛 的 颜色 ， 但 不 知道 自 
己 的 (别人 也 不 可 以 告知 )。 此 外 ， 他 们 不 知道 岛 上 到 底 有 多 少 人 是 蓝 眼睛 的 ， 只 知道 至 少 
有 一 个 人 的 眼睛 是 蓝 色 的 。 所 有 蓝 眼 睛 的 人 要 花 几 天 才能 离开 这 个 岛 ” (第 177 页 ) 

6.5 有 栋 建 筑 物 高 100 层 。 若 从 第 N 层 或 更 高 的 楼 层 扔 下 来 ,鸡蛋 就 会 破 掉 。 若 从 第 N 层 以 下 的 楼 
层 扔 下 来 则 不 会 破 掉 。 给 你 2 个 鸡蛋 , 请 找 出 N, 并 要 求 最 差 情 况 下 扔 鸡蛋 的 次 数 为 最 少 。( 第 
178 页 ) 

6.6 走廊 上 有 100 个 关上 的 储 物 柜 。 有 个 人 先是 将 100 个 柜子 全 都 打开 。 接着, 每 数 两 个 柜子 关上 
一 个 。 然后, 在 第 三 轮 时 ,再 每 隔 两 个 就 切换 第 三 个 柜子 的 开关 状态 (也 就 是 将 关上 的 柜子 
打开 ,将 打开 的 关上 )。 照 此 规律 反复 操作 100 次 ,在 第 ; 轮 ， 这 个 人 会 每 数 ;个 就 切换 第 ;个 柜 
子 的 状态 。 当 第 100 轮 经 过 走廊 时 ， 只 切换 第 100 个 柜子 的 开关 状态 ， 此 时 有 几 个 柜子 是 开 着 
的 ? (第 179 页 ) 


8.7 ”数学 与 概率 


在 面试 时 碰 到 的 许多 数学 问题 ， 其 中 很 多 看 起 来 像 是 智力 题 ， 其 实 大 都 可 以 运用 逻辑 、 有 系 
统 地 解决 。 这 些 问 题 通常 都 以 数学 或 计算 机 科学 为 基础 , 运用 这 些 知识 可 以 解决 问题 或 检验 答案 
对 错 。 本 节 将 介绍 那些 关系 最 紧密 的 数学 概念 。 

1. 素数 

大 家 应 该 都 知道 ， 每 一 个 数 都 可 以 分 解 成 素数 为 乘积 。 例 如 : 

84=2°*31*50*7 1 *]11 * 13°* 1]17 *... 

注意 其 中 不 少 素 数 的 指数 为 零 。 

@ 整除 

上 面 的 素数 定理 指出 ， 要 想 以 x 整除 y ( 写作 xy， 或 mod(y, x) = 0 )，x 素 因子 分 解 的 所 有 素数 
必须 出 现在 y 的 素 因子 分 解 中 。 具 体 如 下 : 


今 x=20* 3 * S22 * J * ]1 1 * Eo 
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令 y = 2 米 3 全 米 5 米 7 米 1]1 妈 ”| 
车 xy， 则 让 <= 友 对 所 有 i 都 成 立 。 
实际 上 ，x 和 y 的 最 大 公约 数 为 : 


gcd(x y)= 2min00， Kk0) 六 3minUl， ,kl) 米 Smina02， Kk2) x 


x 和 y 的 最 小 公 倍数 为 : 


lem(x, ») = 2max00 40) * 3max0L Kl) ¥ Smax02 2) * ,,. 
3 


下 面 先 做 一 个 趣味 练习 ， 想 


gcd 米 lcm 本 2min00， Kk0) x* 2max00， 


一 想 ， 将 gcd 与 Iaem 相 乘 ， 结 果 是 什么 ? 


K0) x* 3min0l kl1) >* 3max0], ,kl) *k 。。。 


= 2minV0， K0) + max(10, k0) x* 3minUl， Kl)+ max(l, kl) x 。。。 


一 21/0+ 如 六 311+ 人 如 来 


一 2]0 2 如 3 米 3 位 #* 人 





1 boolean primeNaive(int n) { 
2 if (n < 2) { 

3 return false; 

4 } 

5 for (int i = 2; i < ni i++) { 
6 if (n % i == 60) { 

7 return false; 

8 } 

9 } 

16 return true; 

11 } 








下 面 有 一 人 处 很 小 但 重要 的 改动 :只 需 迭 代 至 的 平方 根 即 可 。 


1 boolean primeSlightlyBetter(int n) { 


2 if (n < 2) { 

3 return false; 

4 } 

5 int sqrt = (int) Math. 
6 for (int i = 2; i <= 
7 if (n % i == 6) ret 
8 } 

9 return true; 


16 } 


sqrt(n); 
sqrt; i++) { 
urn false; 





性 
人 问题 腿 常 见 ， 有 必要 特别 说 明 一 下 。 最 原始 的 做 法 是 从 2 到 n-1 进 


迭代 , 每 次 迭代 都 检 


使 用 sqrt 就 够 了 ， 因 为 每 个 可 以 整除 n 的 数 a， 都 有 个 补 数 b»， 且 a * bp = n。 若 a > sqrt， 
n 族 因此 ， 当 a 大 于 sqrt 时 ， 就 不 需要 检查 了 ， 因 为 已 经 用 b 检 


则 2 < sqrt (因为 sqrt * sqrt = 
查 过 了 。 




















当然 , 在 现实 中 , 我 们 真正 要 做 的 只 是 检查 n 可 否 被 素数 整除 。3 


of Eratosthenes ) 就 派 上 用 场 了 。 
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这 时 埃 拉 托 











斯 特 尼 得法 (Sieve 
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@ 生成 素数 序列 : 埃 拉 托 斯 特 尼 得 法 

埃 拉 托 斯 特 尼 得 法 能 够 非常 高 效 地 生成 素数 序列 ， 原 理 是 剔除 所 有 可 被 素数 整除 的 非 素数 。 

一 开始 列 出 到 max 为 止 的 所 有 数字 。 首 先 , 划 掉 所 有 可 被 2 整除 的 数 (2 保留 )。 然后， 找到 下 
es 被 划 掉 的 数 )， 并 划 掉 所 有 可 被 它 整 除 的 数 。 划 掉 所 有 可 被 2、3、5、 
、11 等 素数 整除 的 数 ， 最 终 可 得 到 2 到 max 之 间 的 素数 序列 。 

下 面 是 埃 拉 托 斯 特 尼 筛 法 的 实现 代码 。 


























1 boolean[] sieveOfEratosthenes(int max) { 
2 boolean[] flags = new boolean[max + 1]; 
3 int count = @; 

4 

5 init(flags); // 将 flags 中 8、1 元 素 除 外 的 所 有 元 素 设 为 true 
6 int prime = 2; 

水 

8 while (prime <= max) { 

9 /* 划 掉 余下 为 prime 倍 数 的 数字 */ 

16 crossOff(flags，prime); 

11 

12 /* 找 出 下 一 个 为 true 的 值 */ 

13 prime = getNextprime(flags, prime); 
14 

15. if (prime >= flags.length) { 

16 break; 

17 } 

18 } 

19 

26 return flags; 

21 } 

22 


23 void crossoff(boolean[] flags, int prime) { 

24 /* 划 掉 余下 为 prime 倍 数 的 数字 ， 我 们 可 以 从 

25 * (prime*prime) 开 始 ， 因 为 如 果 k * prime 且 

26 * k < prime， 这 个 值 早 就 在 之 前 的 选 代 里 

27 * 被 划 掉 了 。 */ 

28 for (int i = prime * prime; i < flags.length; i += prime) { 


29 flags[i] = false; 

36 

31 } 

32 

33 int getNextPrime(boolean[] flags, int prime) { 
34 int next = prime + 1; 

35 while (next < flags.length && !flags[next]) { 
36 next++; 

37 } 

38 return next; 

39 } 





当然 ,在 上 面 的 代码 中 ,还 有 一 些 地 方 可 以 优化 ， 比 如 ,可 以 只 将 奇数 放 进 数组 ， 所 需 空间 
即 可 减 半 。 

2. 概率 

概率 可 以 很 复杂 ， 还 好 它 是 建立 在 若干 基本 定理 之 上 ， 而 这 些 定 理 可 以 逻辑 推导 得 出 。 
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下 面 用 韦 恩 图 ( Venn diagram ) 来 表示 两 个 事件 A 和 事件 B。 两 个 圆圈 的 区 域 分 别 代表 事件 发 
生 的 概率 ， 重 关 区 域 代表 事件 A 与 事件 B 都 发 生 的 概率 ( {A 与 B 都 发 生 } )。 


@ A 与 B 都 发 生 的 概率 

假设 你 朝 上 面 的 韦 恩 图 扔 飞镖 ,命中 A 和 B 重 羞 区域 的 概率 有 多 大 ?如 果 你 知道 命中 A 的 概 
率 ， 还 知道 A 区 域 那 一 块 也 在 B 区 域 中 的 百分比 (也 即 , 命中 A 的 同时 也 在 B 区 域 中 的 概率 )， 即 
可 用 下 面 的 算式 计算 命中 概率 : 

P(A 与 B 都 发 生 ) = P(B 发 生 ， 在 A 发 生 的 情况 下 ) * P(A 发 生 ) 

举 个 例子 ,假设 要 在 1 到 10 ( 含 ) 之 间 挑 选 一 个 数 。 挑 中 一 个 偶数 且 这 个 数 在 1 到 5 之 间 的 概 
率 有 多 大 ? 挑 中 的 数 在 1 到 5 之 间 的 概率 为 30%， 而 在 1 到 5 之 间 的 数 为 偶数 的 概率 为 40%。 因 此 ， 
两 者 同时 发 生 的 概率 为 : 

P(x 为 偶数 且 x <= 5) 

= P(x 为 偶数 ， 在 x <= 5 的 情况 下 ) * P(x <= 5) 
= (2/5) * (1/2) 
= 1/5 

@ A 或 B 发 生 的 概率 

现在 ,我 们 又 想 知 道 飞 镖 命中 A 或 B 的 概率 有 多 大 。 如 果 知 道 单独 命中 A 或 B 的 概率 ， 以 及 命 
中 两 者 重 铸 区 域 的 概率 ， 那么 ， 可 以 用 下 面 的 算式 表示 命中 概率 : 

P(A 或 发生 ) = P(A 发 生 ) + P(B 发 生 ) - P(A 与 B 都 发 生 ) 

这 也 很 合乎 逻辑 。 只 是 简单 地 把 两 个 区 域 加 起 来 , 重合 区 域 就 会 被 计 入 两 次 。 我 们 要 减 掉 一 
次 重 到 区 域 ， 再 次 用 韦 恩 图 表示 : 


举 个 例子 ,假定 我 们 要 在 1 到 10 ( 含 ) 之 间 挑 选 一 个 数 。 挑 中 的 数 为 偶数 或 这 个 数 在 1 到 5 之 
间 的 概率 有 多 大 ? 显然 ， 挑 中 一 个 偶数 的 概率 为 50%， 挑 中 的 数 在 1 到 5 之 间 的 概率 为 50%。 两 者 
同时 发 生 的 概率 为 20%， 因 此 前 面 提 到 的 概率 为 : 
P(x 为 偶数 或 x <=5) 
= P(x 为 偶数 ) + P(x <= 5) - P(x 为 偶数 有 x <= 5) 
= (1/2) + (1/2) - (1/5) 
= 4/5 
掌握 上 述 原 理 后 ， 理 解 独立 事件 和 互 斥 事件 的 特殊 规则 就 要 容易 多 了 。 
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@ 独立 


知 A 与 B 相 互 独立 (也 即 ， 一 个 事件 的 发 生 ， 推 不 出 另 一 个 事件 的 发 生 )， 那 么 ，P(A 与 B 都 发 


生 ) = P(A) P(B)。 这 条 规则 直接 推导 自 P(B 发 生 ， 在 A 发 生 的 情况 下 ) = P(B) ， 因 为 A 跟 B 没 关系 。 








@ 互 斤 
若 A 与 B 互 斥 (也 即 ， 若 一 个 事件 发 生 ， 则 另 一 个 事件 就 不 可 能 发 生 )， 则 P(A 或 B 发 生 ) = 








P(A) + P(B)。 这 是 因为 P(A 与 B 都 发 生 ) = 6， 所 以 ， 删 除了 之 前 P(A 或 B 发 生 ) 算 式 中 的 P(A 与 B 
都 发 生 ) 一 项 。 





奇怪 的 是 ,许多 人 都 会 搞 混 独立 和 互 斥 的 概念 。 其 实 两 者 完全 不 同 。 实 际 上 ， 两 个 事件 不 可 




















能 同时 是 独立 的 又 是 互 斥 的 〈《 只 要 两 者 概率 都 大 于 零 )。 为 什么 ”因为 互 斥 意味 着 一 个 事件 发 生 








了 , 男 一 个 事件 就 不 可 能 发 生 。 而 独立 是 指 一 个 事件 的 发 生 跟 允 一 个 事件 的 发 生 毫 无 关联 。 因此 ， 
只 要 两 个 事件 发 生 的 概率 不 为 零 ， 它 们 就 不 可 能 既 互 斥 又 独立 。 





若 一 个 或 两 个 事件 的 概率 为 零 (也 就 是 不 可 能 发 生 )， 那 么 这 两 个 事件 同时 既 独 立 又 互 斥 。 





这 很 容易 直接 应 用 独立 和 互 斥 的 定义 〈 等 式 ) 证 明 出 来 。 








注意 事项 

(1) 小 心 ，float 和 double 的 精度 有 别 。 

(2) 不 要 假设 某 个 值 ( 比如 一 条 线 的 斜率 ) 为 int 型 ， 除 非 已 明确 告知 这 个 值 为 int 型 。 

(3) 除非 另 有 说 明 ， 和 否则 不 要 假定 多 个 事件 是 独立 的 (或 互 斥 的 )。 因 此 ， 切 忌 育 目 将 概率 相 





乘 或 相 加 。 





面试 题目 





7. 


一 


7.2 


7.3 
7.4 
7.5 


7.6 
7.7 


有 个 篮球 框 ， 下 面 两 种 玩法 可 任 选 一 种 。 

玩法 1: 一 次 出 手机 会 ， 投 篮 命 中 得 分 。 

玩法 2: 三 次 出 手机 会 ， 必 须 投 中 两 次 。 

如 果 p 是 某 次 投篮 命中 的 概率 ， 则 p 的 值 为 多 少时 ， 才 会 选择 玩法 1 或 玩法 2? (第 179 页 ) 
三 角形 的 三 个 顶点 上 各 有 一 只 蚂蚁 。 如 果 蚂 蚊 开 始 沿 着 三 角形 的 边 朴 行 , 两 只 或 三 只 蚂蚁 撞 
在 一 起 的 概率 有 多 大 ? 假定 每 只 蚂蚁 会 随机 选 一 个 方向 , 每 个 方向 被 选 到 的 几率 相等 ,而 且 
三 只 蚂蚁 的 爬行 速度 相同 。 

类 似 问 题 : 在 n 个 顶点 的 多 边 形 上 有 nn 只 蚂蚁 ， 求 出 这 些 蚂 蚁 发 生 碰 撞 的 概率 。( 第 180 页 ) 
给 定 直角 坐标 系 上 的 两 条 线 ， 确 定 这 两 条 线 会 不 会 相交 。( 第 181 页 ) 

编写 方法 ， 实 现 整数 的 乘法 、 减 法 和 除法 运算 。 只 允许 使 用 加 号 。( 第 182 页 ) 

在 二 维 平面 上 ,有 两 个 正方 形 , 请 找 出 一 条 直线 ， 能够 将 这 两 个 正方 形 对 半分 。 假定 正方形 
的 上 下 两 条 边 与 x 轴 平 行 。( 第 184 页 ) 

在 二 维 平面 上 ， 有 一 些 点 ， 请 找 出 经 过 点 数 最 多 的 那 条 线 。( 第 186 页 ) 

有 些 数 的 素 因 子 只 有 3、5、7， 请 设计 一 个 算法 ， 找 出 其 中 第 个 数 。( 第 188 页 ) 


参考 问题 ， 中 等 难题 (#17.11 ) ; 高 难度 题 (#18.2 ) 。 
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8.8 面向 对 象 设计 


面向 对 象 设计 问题 要 求 求职 者 设计 出 类 和 方法 ， 以 实现 技术 问题 或 描述 真实 生活 中 的 对 象 。 
这 类 问题 可 以 让 面试 官 洞悉 你 的 编码 风格 一 一 至 少 被 认为 如 此 。 

这 些 问题 并 不 那么 着 重 于 设计 模式 , 而 是 意 在 考察 你 是 否 懂 得 如 何 打 造 优 雅 、 容 易 维护 的 面 
回 对 象 代码 。 知 在 这 类 问题 上 表现 不 佳 ， 面 试 可 能 会 亮 起 红 灯 。 

1. 如 何 解 答 面向 对 象 设计 问题 

对 于 面向 对 象 设计 问题 ， 要 设计 的 对 象 可 能 是 真实 世界 的 东西 ,也 可 能 是 某 个 技术 任务 ,不 
论 如 何 ， 我 们 都 能 以 类 似 的 途径 解决 。 以 下 解 题 思路 适用 于 很 多 问题 。 

@ 步骤 1: 处 理 不 明确 的 地 方 

面向 对 象 设计 ( OOD ) 问题 往往 会 故意 放 些 烟 幕 弹 , 意 在 检验 你 是 武断 腾 测 , 还 是 提出 问题 
以 厘清 问题 。 毕 竞 ， 开发 人 员 要 是 没 弄 清楚 自己 要 开发 什么 , 就 直接 挽 起 袖子 开始 编码 ， 只 会 浪 
费 公 司 的 财力 物力 ， 还 可 能 造成 更 严重 的 后 果 。 

磁 到 面向 对 象 设计 间 题 时 ， 你 应 该 先 问 清楚 ， 谁 是 使 用 者 、 他 们 将 如 何 使 用 。 对 某 些 问 题 ， 
你 甚至 还 要 问 清楚 “5W1H”， 也 就 是 Who ( 谁 )、What (什么 )、Where (哪里 )、When ( 何 时 小 
Why (为 什么 )、How (如何 )。 

举 个 例子 , 假设 面试 官 让 你 描述 咖啡 机 的 面向 对 象 设计 。 这 个 问题 看 似 简单 明了 , 其 实 不 然 。 

这 台 咖 啡 机 可 能 是 一 球 工 业 型 机 器 ,设计 用 来 放 在 大 餐厅 里 , 每 小 时 要 服务 几 百 位 顾客 , 还 
要 能 制作 10 种 不 同 口味 的 咖啡 。 又 或 者 , 它 可 能 是 设计 给 老年 人 使 用 的 简易 咖啡 机 ， 只 要 能 制作 
简单 的 黑 咖 啡 就 行 。 这 些 用 例 将 大 大 影响 你 的 设计 。 

@ 步骤 2: 定义 核心 对 象 

了 解 我 们 要 设计 的 东西 后 ， 接 下 来 就 该 思考 系统 的 “核心 对 象 ” 了 。 比 如 ,假设 要 为 一 家 和 餐 
馆 进行 面向 对 象 设计 。 那 么 ， 核 心 对 象 可 能 包括 餐桌 ( Table )、 顾 客 ( Guest )、 宴 席 ( Party )、 
订单 (Order )、 餐 点 (Meal )、 员 工 (Employee )、 服 务 员 ( Server ) 和 领班 ( Host )。 

@ 步骤 3: 分 析 对 象 关 系 

定义 出 核心 对 象 之 后 ,， 接 下 来 要 分 析 这 些 对 象 之 间 的 关系 。 其 中 , 哪些 对 象 是 其 他 对 象 的 数 
据 成 员 ? 哪个 对 象 继承 自 别 的 对 象 ?” 对 象 之 间 是 多 对 多 的 关系 ， 还 是 一 对 多 的 关系 ? 
比如 ， 在 处 理 和 餐馆 问题 时 ， 我 们 可 能 会 想到 以 下 设计 。 

口 宴席 包括 很 多 顾客 。 

口 服务 员 和 领班 都 继承 自 员 工 。 

口 每 一 张 餐 更 对 应 一 个 宰 席 ， 但 每 个 宰 席 可 能 拥有 多 张 餐桌 。 
口 每 家 餐馆 有 一 个 领班 。 

分 析 对 象 关系 一 定 要 非常 小 心 一 一 我 们 经 党 会 作出 错误 假设 。 比 如 , 哪怕 是 一 张 餐桌 也 可 能 
包含 多 个 “宴席 ”( 在 热门 餐馆 里 ,“ 拼 桌 ” 很 常见 )。 进 行 设计 时 ， 你 应 该 跟 面试 官 探 讨 一 下 ， 
了 解 你 的 设计 要 做 到 多 通用 。 
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@ 步骤 4: 研究 对 象 的 动作 

到 这 一 步 , 你 的 面向 对 象 设计 应 该 初 具 锥 形 了 。 接 下 来 , 该 想 想 对 象 可 执行 的 关键 动作 ， 以 
及 对 象 之 间 的 关联 。 你 可 能 会 发 现 自己 遗漏 了 某 些 对 象 ， 这 时 就 需要 补 全 并 更 新 设计 。 

例如 ， 一个“ 宴席” 对象 ( 由 一 群 顾客 组 成 ) 走 进 了 “餐馆 ”， 一 位 “顾客 ” 找 “ 领 班 ”要 
求 一 张 “和 餐桌” 。“ 和 领班” 开始 查验 “预订 ”( Reservation )， 若 找到 记录 ， 便 将 “宴席 ”对 象 领 
到 “和 餐桌” 前。 否则 ,“ 宴 席 ” 对 象 就 要 排 在 列表 末尾 。 等 到 其 他 “宴席 ”对 象 离开 后 ， 有 “和 餐 
桌 ” 空 出 来 ， 就 可 以 分 配给 列表 中 的 “宴席 ”对 象 。 

2. 设计 模式 

因为 面试 官 想 要 考察 的 是 你 的 能 力 而 不 是 知识 ,大 部 分 面试 都 不 会 考 设计 模式 。 不 过 , 掌握 
单 例 设计 模式 (Singleton ) 和 工厂 方法 ( Factory Method ) 设计 模式 对 面试 来 说 特别 有 用 ， 所 以 ， 
接 下 来 我 们 会 作 简单 介绍 。 

设计 模式 太 多 了 ， 限 于 篇 幅 ,， 没 办 法 在 本 书 中 一 一 探讨 。 你 可 以 挑 本 专门 讨论 这 个 主题 的 书 
来 研读 ， 这 对 提高 你 的 软件 工程 技能 会 大 有 神 益 。 

@ 单 例 设计 模式 

单 例 设 计 模 式 确保 一 个 类 只 有 一 个 实例 ,并 且 只 能 通过 类 内 部 方法 访问 此 实例 。 当 你 有 个 “全 
局 ”对 象 ， 并 且 只 会 有 一 个 这 种 实例 时 ， 这 个 模式 特别 好 用 。 比 如 ， 在 实现 “餐馆 ”时 ,我 们 可 
能 想 让 它 只 有 一 个 “和 餐馆 ”实例 。 












































1 public class Restaurant { 

2 private static Restaurant _instance = null; 
3 protected Restaurant() { ... } 

4 public static Restaurant getInstance() { 

5 if (_instance == null) { 

6 _instance = new Restaurant(); 

7 } 

8 return _instance; 

9 } 

10 } 


@ 工厂 方法 设计 模式 

工厂 方法 提供 接口 以 创建 某 个 类 的 实例 , 由 子 类 决定 实例 化 哪个 类 。 实 现时 ， 你 可 以 将 创建 
器 类 〈 Creator ) 设计 为 抽象 类 型 ， 不 为 工厂 方法 提供 具体 实现 ; 或 者 ， 创 建 器 类 是 实体 类 ， 为 工 
三 方法 提供 具体 实现 。 在 这 种 情况 下 ， 工 三 方法 需要 传人 参数 ， 代 表 该 实例 化 哪个 类 。 





public class CardGame { 
public static CardGame createCardGame(GameType type) { 
if (type == GameType.Poker) { 
return new PokerGame(); 
} else if (type == GameType.BlackJack) { 
return new BlackJackGame(); 


return null; 
: 


1 
2 
3 
4 
5 
6 
7 
8 
9 
16 } 
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面试 题目 
8.1 请 设计 用 于 通用 扑克 牌 的 数据 结构 。 并 说 明 你 会 如 何 创 建 该 数据 结构 的 子 类 , 实现 “二 十 一 


点 ”游戏 。( 第 192 页 ) 

8.2 设想 你 有 个 呼叫 中 心 ， 员 工分 成 三 个 层级 : 接线 员 、 主 管 和 经 理 。 客 户 来 电 会 先 分 配给 有 空 
的 接线 员 。 若 接线 员 处 理 不 了 ， 就 必须 将 来 电 往 上 转 给 主管 。 若 主管 没 空 或 是 无 法 处 理 ， 则 

将 来 电 往 上 转 给 经 理 。 请 设计 这 个 问题 的 类 和 数据 结构 ， 并 实现 一 个 dispatchcall() 方 法 ， 

将 客户 来 电 分 配给 第 一 个 有 空 的 员工 。( 第 195 页 ) 

8.3 运用 面向 对 象 原 则 ， 设 计 一 款 音乐 点 唱机 。( 第 198 页 ) 

8.4 运用 面向 对 象 原则 ， 设 计 一 个 停车 场 。( 第 200 页 ) 

8.5 请 设计 在 线 图 书 阅 读 器 系统 的 数据 结构 。( 第 203 页 ) 

8.6 实现 一 个 拼图 程序 。 设 计 相关 数据 结构 并 提供 一 种 拼图 算法 。 假 设 你 有 一 个 fitswith 方 法 ， 
传人 两 块 拼图 ， 阁 两 块 拼 图 能 拼 在 一 起 ， 则 返回 true。( 第 207 页 ) 

8.7 请 描述 该 如 何 设计 一 个 聊天 服务 器 。 要求 给 出 各 种 后 台 组 件 、 类 和 方法 的 细节 ,并 说 明 其 中 
最 难 解决 的 问题 会 是 什么 。( 第 210 页 ) 

8.8“ 奥 赛 罗 棋 ”( 黑白 棋 ) 的 玩法 如 下 : 每 一 枚 棋子 的 一 面 为 白 ,一 面 为 黑 。 游 戏 双 方 各 执 黑 、 
白 棋子 对 决 ， 当 一 枚 棋子 的 左右 或 上 下 同时 被 对 方 棋子 夹 住 ， 这 枚 棋子 就 算是 被 吃 掉 了 ， 随 
即 翻 面 为 对 方 棋子 的 颜色 。 轮 到 你 落 子 时 ， 必 须 至 少 吃 掉 对 方 一 枚 模子 。 任 意 一 方 无 子 可 落 
时 , 游戏 即 告 结束 。 最后, 棋盘 上 棋子 较 多 的 一 方 获胜 。 请 运用 面向 对 象 设 计 方 法 , 实现 “ 奥 
赛 罗 棋 ”。( 第 214 页 ) 

8.9 设计 一 种 内 存 文件 系统 (in-memory file system ) 的 数据 结构 和 算法 ， 并 说 明 具体 做 法 。 如 有 
可 行 ， 请 用 代码 举例 说 明 。( 第 217 页 ) 

8.10 设计 并 实现 一 个 散 列 表 ， 使 用 链接 ( 即 链表 ) 处 理 碰撞 冲突 。( 第 219 页 ) 


参考 问题 : 线程 与 锁 (#16.3 ) 。 
8.9 递归 和 动态 规划 
尽管 递归 问题 五 花 八 门 , 但 题 型 大 都 类 似 。 一 个 问题 是 不 是 递归 的 , 就 看 它 能 不 能 分 解 为 子 


























































































































问题 进行 求解 。 
当 你 听 到 问题 是 这 么 开头 的 :“ 设 计 一 个 算法 , 计算 第 n 个 ……”,“ 编 写 代码 列 出 前 4 个 ……”， 


“实现 一 个 方法 ,计算 所 有 ……” 等 等 ， 那么， 这 基本 上 就 是 一 个 递归 问题 。 

熟 能 生 巧 ! 练习 的 越 多 ， 就 越 容易 识别 递归 问题 。 

1. 解决 之 道 

递归 的 解法 ， 根 据 定 义 ， 就 是 从 较 小 的 子 问题 逐渐 逼近 原始 问题 。 很 多 时 候 ， 只 要 在 六 -1 
的 解法 中 加 入 、 移 除 某 些 东西 或 者 稍 作 修 改 就 能 算出 /0D)。 而 在 其 他 情况 下 ， 答 案 可 能 更 为 复杂 。 

你 应 该 双管齐下 , 自 下 而 上 和 自 上 而 下 两 种 递归 解法 都 要 考虑 。 简 单 构造 法 对 递归 问题 就 很 
奏效 。 
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@ 自 下 而 上 的 递归 

自 下 而 上 的 递归 往往 最 为 直观 。 首 先 要 知道 如 何 解 决 简单 情况 下 的 问题 ， 比 如 ， 只 有 一 个 元 
素 的 列表 ， 找 出 有 两 个 、 三 个 元 素 的 列表 的 解法 ,， 依 此 类 推 。 这 种 解法 的 关键 在 于 ， 如 何 从 先前 
解 出 来 的 答案 ， 构 建 出 后 续 情 况 的 答案 。 

@ 自 上 而 下 的 递归 

自 上 而 下 的 递归 可 能 比较 复杂 , 不 过 对 某 些 问题 很 有 必要 。 遇 到 这 类 问题 时 ,我 们 要 思考 如 
何 才 能 将 情况 NT 下 的 问题 分 解 成 多 个 子 问 题 。 同 时 要 注意 子 问题 是 否 重 合 了 。 

2. 动态 规划 

在 面试 中 ， 动 态 规 划 (Dynamic programming，DP ) 问题 很 少 间 及 ， 原 因 很 简单 ， 短 短 45 分 
钟 的 面试 要 解决 这 类 问题 实在 太 难 了 。 就 算是 出 色 的 求职 者 , 面 对 这 类 问题 通常 也 难 有 上 佳 表 现 ， 
因此 动态 规划 问题 不 适合 用 来 评估 求职 者 。 

要 是 不 走运 , 碰 到 了 动态 规划 问题 , 你 可 以 用 跟 递 归 问 题 差不多 的 解决 方法 来 处 理 。 区 别 在 
于 ， 中 间 结 果 要 “缓存 ”起 来 ， 以 备 后 续 使 用 。 

@ 动态 规划 法 简单 示例 : 斐 波 那 契 数列 

下 面 举 个 动态 规划 法 的 简单 例子 。 想 象 一 下 ,面试 官 要 求 你 实现 一 个 程序 , 生成 斐 波 那 契 数 
列 的 第 xz 项 数字 。 听 起 来 很 简单 ， 对 吧 ? 



























































1 int fibonacci(int i) { 

之 if (i == 6) return 6; 

3 if (i == 1) return 1; 

4 return fibonacci(i - 1) + fibonacci(i - 2); 

5 } 

这 个 函数 要 用 多 长 时 间 执 行 ? 计算 斐 波 那 契 数列 第 z 项 要 先 算出 前 面 的 盖 1 项 。 而 每 次 函数 调 
用 又 包含 两 次 递归 调用 ， 也 就 是 说 ， 执 行 时 间 为 CC2)。 下 面 的 图 表 显 示 在 普通 台式 机 上 的 执行 
结果 ， 执 行 时 间 呈 指数 级 上 升 。 














0 1 20 30 4 em 


生成 第 V 项 斐 波 那 契 数 所 需 秒 数 


只 要 对 上 面 的 函数 稍 作 修改 ， 就 可 以 将 时 间 复 杂 度 优化 为 OW)。 具 体 做 法 就 是 将 每 次 调用 
fibonacci(i) 的 结果 “缓存 ”起 来 。 
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1 int[] fib = new int[max]; 

2 int fibonacci(int i) { 

3 if (i == 6) return 6; 

if (i == 1) return 1; 

if (fib[i] != 6) return fib[i]; // 返回 先前 缓存 的 结果 
fib[i] = fibonacci(i - 1) + fibonacci(i - 2); // 缓存 结果 
return fib[i]; 


} 

在 普通 电脑 上 , 之 前 的 递归 版 本 生成 第 50 项 斐 波 那 契 数 用 时 可 能 超过 一 分 钟 ， 而 动态 规划 方 
法 只 需 几 毫秒 就 能 产生 第 10 000 项 斐 波 那 契 数 。 当 然 ， 若 采用 上 面 这 段 代 码 ，int 型 变量 很 快 就 
会 溢出 。 

如 你 所 见 ， 动态 规划 法 没什么 好 怕 的 。 只 不 过 要 缓存 中 间 结 果 ， 比 递归 稍稍 复杂 些 。 解 决 这 
类 问题 ， 有 个 好 办 法 就 是 先 以 一 般 的 递归 法 实现 ， 然 后 添加 缓存 部 分 。 

3. 递归 和 迭代 解法 

递归 算法 的 空间 效率 很 低 。 每 次 递归 调用 都 会 在 栈 上 增加 一 层 ， 也 就 是 说 ， 若 算法 包含 O(0D) 
次 递归 调用 ， 就 要 使 用 O(n) 内 存 。 不 得 了 1! 

所 有 的 递归 代码 都 能 改 为 迭代 式 的 实现 , 尽管 有 时 候 这 么 做 代码 会 复杂 得 多 。 在 一 头 扎 入 递 
归 代 码 之 前 ， 先 问 问 自己 用 和 迭代 方式 实现 会 有 多 难 ， 并 与 面试 官 讨论 不 同 解法 的 优 劣 差异 。 


o vi 人 







































































面试 题目 





9.1 有 个 小 孩 正在 上 楼 梯 ， 楼 梯 有 7z 阶 台阶 ,小 孩 一 次 可 以 上 1 阶 、2 阶 或 3 阶 。 实 现 一 个 方法 ,， 计 
算 小 孩 有 多 少 种 上 楼 梯 的 方式 。( 第 221 页 ) 
9.2 设想 有 个 机 器 人 坐 在 Xx7 网 格 的 左上 角 ， 只 能 向 右 、 向 下 移动 。 机 器 人 从 (0,0) 到 (%, 广 有 多 少 
种 走 法 ? 
进 阶 
假设 有 些 点 为 “禁区 ”,， 机 器 人 不 能 踏足 。 设 计 一 种 算法 ， 找 出 一 条 路 径 ， 让 机 器 人 从 左上 
角 移 动 到 右 下 角 ( 第 222 页 )。 
9.3 在 数组 A[8...n-1] 中 ， 有 所 谓 的 魔术 索引 ， 满 足 条 件 A[i] =i。 给 定 一 个 有 序 整数 数组 ， 元 
素 值 各 不 相同 ， 编 写 一 个 方法 ， 在 数组 A 中 找 出 一 个 魔术 索引 ， 若 存在 的 话 。 
进 阶 
如 果 数 组 元 素 有 重复 值 ， 又 该 如 何 处 理 ?” (第 224 页 ) 
9.4 编写 一 个 方法 ， 返 回 某 集合 的 所 有 子 集 。( 第 226 页 ) 
9.5 编写 一 个 方法 ， 确 定 某 字 符 串 的 所 有 排列 组 合 。( 第 229 页 ) 
9.6 实现 一 种 算法 ， 打 印 z 对 括号 的 全 部 有 效 组 合 〈 即 左右 括号 正确 配对 )。 
示例 
输入 : 3 
输出 : ((()))，(()())，(())()，()(())，()()()《〈 第 230 页 ) 
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9.7 编写 函数 ， 实 现 许 多 图 片 编 辑 软 件 都 支持 的 “填充 颜色 ”功能 。 给 定 一 个 屏幕 ( 以 二 维 数 组 

表示 ， 元 素 为 颜色 值 )、 一 个 点 和 一 个 新 的 颜色 值 ， 将 新 颜色 值 填 人 这 个 点 的 周围 区 域 ， 直 

到 原来 的 颜色 值 全 都 改变 。( 第 232 页 ) 

给 定数 量 不 限 的 硬币 ,币值 为 25 分 、10 分 、5 分 和 1 分 , 编写 代码 计算 n 分 有 几 种 表示 法 。( 第 

232 页 ) 

9.9 设计 一 种 算法 ， 打 印 八 皇后 在 8x8 棋 盘 上 的 各 种 摆 法 ， 其 中 每 个 皇后 都 不 同行 、 不 同 列 ， 也 
不 在 对 角 线 上 。 这 里 的 “对 角 线 ” 指 的 是 所 有 的 对 角 线 , 不 只 是 平分 整个 棋盘 的 那 两 条 对 角 
线 。( 第 234 页 ) 

9.10 给 你 一 堆 n 个 箱子 ， 箱 子 宽 w;、 高 h,、 深 d;。 箱 子 不 能 翻转 ， 将 箱子 堆 起 来 时 ， 下 面 箱子 的 

宽度 、 高 度 和 深度 必须 大 于 上 面 的 箱子 。 实 现 一 个 方法 ， 搭 出 最 高 的 一 堆 箱 子 ， 箱 堆 的 高 

度 为 每 个 箱子 高 度 的 总 和 。( 第 236 页 ) 

给 定 一 个 布尔 表达 式 ， 由 6、1、&、| 和 人 ^ 等 符号 组 成 ， 以 及 一 个 想 要 的 布尔 结果 result， 

实现 一 个 函数 ， 算 出 有 几 种 括号 的 放 法 可 使 该 表达 式 得 出 result 值 。 

示例 

表达 式 : 1^*6|8|1 

期 望 结果 : false(6) 

输出 : 1^((6|6)|1) 和 1^(6|(e6|1)) 两 种 方式 (第 238 页 ) 


参考 问题 链表 ( 雪 .2、 吉 .5、 坟 .7 ) ; 栈 与 队列 (#2.3 ) ; 树 与 图 ( 业 .1、 业 .3、 州 .4、 州 .5、 
#4.7、#4.8、 扒 .9 ) ; 位 操作 (#5.7 ) ; 智力 题 (#6.4 ) ; 排序 与 查找 (#11.5、#11.6、#11.7、#11.8); 
C 和 C++ (#13.7 ) ; 中 等 难题 (#17.13、#17.14 ) ; 高 难度 题 (#18.4、#18.7、#18.12、#18.13 ) 。 


8.10 ”扩展 性 与 存储 限制 


扩展 性 面试 题 看 似 吓人 人， 其实 这 类 问题 算得 上 是 最 简单 的 。 它 们 不 会 暗藏 什么 “陷阱 ”"， 不 
会 有 什么 花招 , 也 不 需要 花哨 的 算法 一 一 至 少 通常 不 会 有 。 你 不 需要 学 习 分 布 式 系统 方面 的 课程 ， 
也 不 必 具 备 系统 设计 的 相关 经 验 。 只 要 稍 加 练习 , 任何 心思 续 密 且 够 聪明 的 软件 工程 师 都 能 轻松 
搞定 这 些 问题 。 

1. 循序 渐进 法 

面试 官 并 不 是 想 考察 你 掌握 了 多 少 系 统 设 计 知 识 ; 实际 上 , 除了 考察 最 基本 的 计算 机 科学 概 
含 ， 面试 官 一 般 不 会 考 具 体 的 知识 点 。 相 反 ,， 他 们 想 要 评估 的 是 你 分 解 棘手 问题 的 能 力 ， 以 及 用 
所 学 知识 解决 问题 的 能 力 。 以 下 这 些 步骤 有 助 于 应 对 大 多 数 系统 设计 问题 。 

@ 步骤 1: 大 胆 假设 
假设 一 台 计 算 机 就 能 装 下 全 部 数据 , 且 存 储 上 没有 任何 限制 。 你 会 如 何 解 决 问题 ? 由 此 得 出 
的 答案 ， 可 以 为 你 最 终 解 决 问题 提供 基本 思路 。 
@ 步骤 2: 切合 实际 
现在 ， 让 我 们 回 到 问题 本 身 。 一 台 计 算 机 究竟 能 装 下 多 少数 据 ,， 拆 分 这 些 数据 会 产生 什么 问 
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题 ? 通常 , 我们 需要 考虑 如 何 合理 拆 分 数据 ,以 及 一 台 计 算 机 需要 不 同 的 数据 片段 时 ， 如 何 得 知 
该 去 哪里 查找 ， 等 等 。 
@ 步骤 3: 解决 问题 
最 后 ， 想 一 想 该 如 何 处 理 步骤 2 发 现 的 问题 。 请 记 住 ， 这 些 解 决 方案 应 该 能 彻底 消除 这 些 问 
题 ,或 至 少 改善 一 下 状况 。 通 常情 况 下 ， 你 可 以 继续 使 用 ( 进行 一 定 修改 ) 步骤 1 描述 的 方法 ， 
但 偶尔 也 需要 改 弦 易 张 ， 从 根本 上 改变 解决 方案 。 
请 注意 ， 和 迭代 法 通常 很 有 用 。 也 就 是 说 ， 等 你 解决 好 步 又 2 发 现 的 问题 ， 可 能 又 会 冒 出 新 间 
题 ， 你 还 要 着 手 处 理 这 些 新 问题 。 
你 的 目标 不 是 重新 设计 公司 耗资 数 百 万 美元 搭建 的 复杂 系统 , 而 是 证 明 你 有 能 力 分 析 和 解决 
问题 。 检 验 自 己 的 解法 ， 四 处 挑 错 并 予以 修正 ， 是 个 向 面试 官 展现 实力 的 不 错 方 法 。 
2. 你 需要 知道 的 ; 信息 、 策 略 与 问题 
@ 典型 系统 
尽管 仍 有 公司 在 使 用 大 型 机 , 可 大 多 数 互 联网 公司 还 是 喜欢 使 用 由 普通 计算 机 互联 组 成 的 大 
型 系统 。 通 常情 况 下 ， 你 可 以 假定 自己 就 是 在 使 用 这 种 系统 。 
面试 之 前 ， 你 最 好 填写 下 面 的 表格 。 利 用 这 张 表 ， 可 以 估算 出 一 台 计算 机 可 存储 多 少数 据 。 
组 一 般 容量 /数值 
硬盘 空间 
内 存 
网 络 传输 延迟 
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@ 拆 分 大 量 的 数据 

尽管 有 时 我 们 可 以 增加 计算 机 的 硬盘 空间 , 不 过 , 难免 会 遇 到 必须 将 数据 拆 分 至 多 台 计 算 机 
的 情形 。 随 之 而 来 的 问题 是 ， 哪 些 数据 要 放 在 哪 一 台 机 器 上 。 下 面 有 几 种 策略 可 供 参考 。 

口 按 出 现 的 顺序 

我 们 可 以 按 出 现 的 顺序 直接 划分 数据 。 也 就 是 说 ， 有 新 数据 进来 时 ， 先 放 进 当前 机 器 ， 填 满 
之 后 ， 再 加 一 台 机 器 。 这 么 做 的 好 处 是 不 会 浪费 资源 。 然 而 ,根据 具体 问题 和 数据 集 的 不 同 , 查 
找 表 可 能 会 变 得 非常 复杂 、 异 常 巨大 。 

口 按 散 列 值 

另 一 种 做 法 是 根据 数据 的 散 列 值 存放 数据 。 具 体 一 点 来 说 ， 我 们 会 采取 以 下 步骤 : (1) 根据 
数据 挑选 某 种 键 ; (2) 利用 散 列 函数 得 到 键 的 散 列 值 ; (3) 将 散 列 值 除 以 机 器 数量 求 得 余数 ; (4) 将 
数据 存储 在 这 个 值 对 应 的 机 器 上 。 也 就 是 说 ， 数 据 会 存放 在 编号 为 #[mod(hash(key)，N)] 的 机 
器 上 。 

这 种 做 法 的 好 处 是 不 用 创建 数据 查找 表 。 每 一 台 计 算 机 自动 掌握 数据 的 存储 位 置 。 然 而 ,这 
也 会 出 问题 , 那 就 是 某 台 机 器 的 数据 可 能 会 多 一 些 , 并 最 终 超出 它 的 存储 容量 。 若 发 生 这 种 情况 ， 
可 以 将 数据 迁移 到 其 他 机 器 上 ， 以 实现 更 好 的 负载 均衡 (但 开销 很 大 )， 或 者 将 这 人 台 机 器 的 数据 
拆 分 到 两 台 机 器 上 《形成 一 组 树 状 结构 的 机 融 )。 
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口 按 实际 值 

按 散 列 值 划分 数据 本 质 上 是 随机 的 ; 数据 代表 的 具体 意义 与 存储 数据 的 机 器 之 间 , 并 不 存在 
任何 关系 。 在 某 些 情况 下 ， 我 们 也 许可 以 利用 数据 所 代表 的 信息 来 降低 系统 延迟 。 

例如 , 假设 你 正在 设计 一 个 社交 网 站 。 虽 然 人 们 的 朋友 会 来 自 世 界 各 地 , 但 实际 上 , 相 比 俄 罗 
斯 普通 公民 , 住 在 墨西哥 的 人 可 能 拥有 更 多 来 自 墨西哥 的 朋友 。 或 许 ,， 我 们 可 以 将 “类 似 ” 数 据 存 
储 在 同一 台 机 器 上 ， 这 样 在 查找 墨西哥 人 的 朋友 时 ， 只 需 访问 较 少 数量 的 机 器 就 能 取得 相关 资料 。 

口 随机 存储 

通常 情况 下 , 我 们 只 是 随机 划分 数据 ,再 实现 一 个 查找 表 以 表明 哪 台 机 器 拥有 哪些 数据 。 虽 
然 这 肯定 需要 一 张 巨 大 的 查找 表 , 但 它 简 化 了 系统 设计 的 某 些 方面 , 使 我 们 得 以 实现 更 好 的 负载 
均衡 。 

3. 示例 : 查找 所 有 包含 某 一 组 词 的 文件 

给 定数 百 万 份 文件 ， 如 何 找 出 所 有 包含 菜 一 组 词 的 文件 ?我 们 不 关心 这 些 词 出 现 的 顺序 , 但 
它们 必须 是 完整 的 单词 。 也 就 是 说 ，“book” 与 “bookkeeper” 不 是 一 回 事 。 

在 着 手 解决 问题 之 前 ， 我 们 需要 考虑 findwords 程 序 只 用 一 次 ， 还 是 要 反复 调用 。 假 设 需要 
多 次 调用 findwords 程 序 来 扫描 这 些 文件 ， 那 么 ,我 们 可 以 接受 预 处 理 的 开销 。 

@ 步骤 ] 

第 一 步 是 先 忘 记 我 们 有 数 以 百 万 计 的 文件 ,假装 只 有 几 十 个 文件 。 在 这 种 情况 下 ,如何 实现 
findwords 呢 ? (提示 : 不 要 急 着 看 下 文 ， 先 试 着 自己 解 解 看 。) 

一 种 方法 是 预 处 理 每 个 文件 , 并 创建 一 个 散 列 表 的 索引 。 这 个 散 列表 会 将 词 映射 到 含有 这 个 
词 的 一 组 文件 。 


“books” -> {doc2, doc3, doc6, doc8} 
“many” -> {docl, doc3, doc7, doc8, doc9} 


若 要 查找 “many books”， 只 需 对 “books” 和 “many” 的 值 进行 交集 运算 ,于 是 得 到 结果 {doc3， 
doc8}。 

@ 步骤 2 

现在 ， 回 到 最 初 的 问题 。 若 有 数 百 万 份 文件 , 会 有 什么 问题 ?首先 , 我 们 可 能 需要 将 文件 分 
散 到 多 台 机 器 上 。 此 外 , 我 们 还 要 考虑 很 多 因素 ， 比 如 要 查找 的 单词 数量 、 在 文件 中 重复 出 现 的 
次 数 等 ， 一 台 机 器 可 能 放 不 下 完整 的 散 列 表 。 假 设 我 们 就 要 按 这 个 限制 进行 设计 。 

文件 分 散 到 多 台 机 器 上 会 引出 以 下 几 个 很 关键 的 关注 点 。 

(1) 如 何 划 分 该 散 列表 ?我 们 可 以 按 关 键 字 划 分 例如， 某 台 机 器 上 存放 有 包含 某 个 单词 的 
全 部 文件 。 或 者 ， 可 以 按 文件 来 划分 ,这样 一 台 机 器 上 只 会 存放 对 应 某 个 关键 字 的 部 分 文件 ， 而 
非 全 部 。 

(2) 一 旦 决定 了 如 何 划 分 数据 ， 我 们 可 能 需要 在 一 台 机 器 上 对 文件 进行 处 理 ， 并 将 结果 推 
送 到 其 他 机 器 上 。 这 个 过 程 会 是 什么 呢 ? ( 注意 : 车 按 文件 划分 散 列 表 ， 这 一 步 可 能 就 没有 
必要 。) 
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(3) 我 们 需要 找到 一 种 方法 获知 哪 台 机 器 拥有 哪些 数据 。 这 个 查找 表 会 是 什么 样 的 ? 又 该 存 
储 在 什么 地 方 ? 

这 只 是 其 中 三 个 ， 可 能 还 会 有 更 多 其 他 的 关注 点 。 

@ 步 又 3 

在 步 又 3 中 ， 我 们 要 找 出 解决 这 些 关 注 点 的 解决 方案 。 其 中 一 种 解法 是 按 字 母 顺序 划分 不 同 
的 关键 字 ， 这 样 ， 每 台 机 器 便 可 以 处 理 一 串 词 。 例 如 ， 从 “after” 直 到 “apple”。 

我 们 可 以 实现 一 个 简单 的 算法 , 按 字母 顺序 遍历 所 有 关键 字 , 并 尽 可 能 多 地 将 数据 存储 在 一 
台 机 器 上 。 当 这 人 台 机 顺 的 空间 被 占 满 之 后 ， 我 们 便 转 到 下 一 台 机 器 。 

这 种 方法 的 优点 是 查找 表 会 比较 小 而 且 简单 〈 因为 它 只 需 包含 一 系列 指定 的 值 )， 每 台 机 器 
可 存储 一 份 查找 表 的 副本 。 然 而 , 不足 之 处 在 于 新 增 文件 或 单词 时 , 我 们 可 能 需要 改变 关键 字 的 
立 置 ， 这 么 做 开销 很 大 。 

为 了 找到 匹配 某 一 组 字符 串 的 所 有 文件 , 我 们 会 先 对 这 一 组 字符 串 进行 排序 , 然后 给 每 一 台 
机 器 发 送 与 字符 对 应 的 查找 请 求 。 例 如 ， 若 待 查 字 符 串 为 “after builds boat amaze banana”， 
一 号 机 器 就 会 接收 到 查找 {“after”，“amaze”} 的 请 求 。 

一 号 机 器 开始 查找 包含 “after” 与 “amaze” 的 文件 ， 并 对 这 些 文件 执行 交集 运算 。 三 号 机 器 
则 处 理 {“banana”，“boat”，“builds”} 这 几 个 关键 字 ， 同 样 也 会 对 文件 进行 交集 运算 。 

最 后 ， 发 送 请 求 的 机 器 再 对 一 号 机 器 及 三 号 机 器 返回 的 结果 作 交 集运 算 。 

下 图 描述 了 整个 过 程 。 
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“after builds boat amaze banana” 

















Machine 1: “after amaze” Machine 3: “builds boat banana” 














“builds” -> doc3, doc4, doc5 
“boat” -> doc2, doc3, doc5 
“banana” -> doc3, doc4, doc5 


“after” -> docl, doc5, doc7 
“amaze” -> doc2, doc5, doc7 




















{doc5, doc7} {doc3, doc5} 


solution = doc5 




















面试 题目 





10.1 假设 你 正在 搭建 某 种 服务 ， 有 多 达 1000 个 客户 端 软件 会 调用 该 服务 ， 取 得 每 天 盘 后 股票 价 
格 信息 (开盘 价 、 收 盘 价 、 最 高 价 与 最 低 价 )。 假设 你 手 里 已 有 这 些 数据 ， 存 储 格式 可 自行 
定义 。 你 会 如 何 设 计 这 套 面向 客户 端的 服务 ， 向 客户 端 软件 提供 信息 ? 你 将 负责 该 服务 的 
研发 、 部 署 、 持 续 监控 和 维护 。 描 述 你 想到 的 各 种 实现 方案 ,以 及 为 何 推 荐 采用 你 的 方案 。 
该 服务 的 实现 技术 可 任 选 ， 此 外 ， 可 以 选用 任何 机 制 向 客户 端 分 发 信息 。( 第 241 页 ) 
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10.2 你 会 如 何 设计 诸如 Facebook 或 LinkedIn 的 超大 型 社交 网 站 ?请 设计 一 种 算法 ,展示 两 个 人 之 
间 的 “连接 关系 ”或 “社交 路 径 ”( 比如 , 我 一 鲍 勃 一 苏 珊 一 杰 森 一 你 )。( 第 243 页 ) 

10.3 给 定 一 个 输入 文件 ， 包 含 40 亿 个 非 负 整数 ， 请 设计 一 种 算法 ， 产 生 一 个 不 在 该 文件 中 的 整 
数 。 假 定 你 有 1GB 内 存 来 完成 这 项 任务 。 
进 阶 
如 果 只 有 10MB 内 存 可 用 ， 该 怎么 办 ? 假定 所 有 值 都 是 唯一 的 。( 第 240 页 ) 

10.4 给 定 一 个 数组 ， 包 含 1 到 入 的 整数 ，N 最 大 为 32 000， 数 组 可 能 含有 重复 的 值 ， 且 的 取 值 不 
定 。 若 只 有 4KB 内 存 可 用 ， 该 如 何 打印 数组 中 所 有 重复 的 元 素 。( 第 248 页 ) 

10.5 如 果 要 设计 一 个 网 络 候 虫 程序 ， 该 怎么 避免 陷入 无 限 循环 ? (第 249 页 ) 

10.6 给 定 100 亿 个 网 址 ， 如 何 检测 出 重复 的 文件 ”这 里 所 谓 的 ”重复 “是 指 两 个 URL 完 全 相同 。 
(第 250 页 ) 

10.7 想象 有 个 Web 服 务 器 ， 实 现 简化 版 搜索 引擎 。 这 套 系 统 有 100 台 机 器 来 响应 搜索 查询 ， 可 能 
会 对 另外 的 机 器 集群 调用 processsearch(string query) 以 得 到 真正 的 结果 。 响 应 查询 请 
求 的 机 器 是 随机 挑选 的 ， 因 此 两 个 同样 的 请 求 不 一 定 由 同一 台 机 器 响应 。 方 法 
processSearch 的 开销 很 大 ， 请 设计 一 种 缓存 机 制 ， 缓 存 最 近 几 次 查询 的 结果 。 当 数据 发 
生变 化 时 ， 务 必 说 明 该 如 何 更 新 缓存 。( 第 251 页 ) 


参考 问题 : 面向 对 象 设计 (#8.7) 。 


8.11 排序 与 查找 


花 时 间 学 习 掌握 常见 的 排序 和 查找 算法 ,回报 巨大 , 很 多 排序 与 查找 问题 , 实际 上 只 是 将 大 
家 熟悉 的 算法 稍 作 修改 而 已 。 因 此 ,处 理 这 类 问题 的 诀 罕 就 是 逐一 考虑 各 种 不 同 的 排序 算法 ,看 
看 哪 一 种 特别 合适 。 

举 个 例子 , 假设 你 被 问 到 如 下 问题 : 给 定 一 个 含有 Person 对 象 日 非 常 大 的 数组 , 请 按 年 龄 从 
小 到 大 对 数组 元 素 进行 排序 。 

根据 题目 ， 有 以 下 两 点 值得 注意 : 

(1) 数组 很 大 ， 所 以 效率 非常 重要 ; 

(2) 根据 年 龄 排序 ， 所 以 这 些 数值 的 范围 较 小 。 

仿 查 各 种 排序 算法 ， 可 能 会 注意 到 “ 桶 排序 ”( 或 称 基数 排序 )， 特 别 适用 于 这 个 问题 。 事实 
上 ， 我们 用 到 的 桶 子 数目 并 不 多 (一 个 年 龄 对 应 一 个 )， 最 终 执行 时 间 为 O(n)。 

1. 常见 的 排序 算法 

学 习 (或 复习 ) 常见 的 排序 算法 是 提升 自身 水 平 的 绝 佳 方式 。 下 面 介 绍 的 五 种 算法 中 ， 归 并 
排序 (Merge Sort )、 快 速 排 序 ( Quick Sort ) 和 基数 排序 (Radix Sort ) 是 面试 中 最 常用 的 三 种 。 

@ 置 泡 排序 | 执行 时 间 : 平均 情况 与 最 差 情况 为 O(n)， 存 储 空间 : O(]) 

冒 泡 排序 ( Bubble Sort ) 是 先 从 数组 第 一 个 元 素 开 始 ， 依 次 比较 相 邻 两 个 数 ， 若 前 者 比 后 者 
大 ， 就 将 两 者 交换 位 置 ， 然 后 处 理 下 一 对 ， 依 此 类 推 , 不 断 扫 描 数 组 ， 直 到 完成 排序 。 
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@ 选择 排序 | 执行 时 间 : 平均 情况 与 最 差 情况 为 O(n)， 存 储 空间 : O(1) 

选择 排序 ( Selection Sort ) 有 点 “小 儿科 ”: 简单 而 低 效 。 我 们 会 线性 逐一 扫描 数组 元 素 ， 从 
中 挑 出 最 小 的 元 素 , 将 它 移 到 最 前 面 ( 也 就 是 与 最 前 面 的 元 素 交 换 )。 然 后， 再 次 线性 扫描 数组 ， 
找到 第 二 小 的 元 素 ， 并 移 到 前 面 。 如 此 反复 ， 直 到 全 部 元 素 各 归 其 位 。 

@ 归并 排序 | 执行 时 间 : 平均 情况 与 最 差 情 况 为 O(n log(0D))， 存 储 空间 : 看 情况 

归并 排序 是 将 数组 分 成 两 半 ， 这 两 半分 别 排序 后 ， 再 归并 在 一 起 。 排 序 基 一 半 时 ,继续 沿用 
同样 的 排序 算法 ， 最终， 你 将 归并 两 个 只 含 一 个 元 素 的 数组 。 这 个 算法 的 重担 都 落 在 “归并 ”的 
部 分 上 。 

在 下 面 的 代码 中 ，merge 方 法 会 将 目标 数组 的 所 有 元 素 拷贝 到 临时 数组 helper 中 ， 并 记 下 数 
组 左 、 右 两 半 的 起 始 位 置 ( helperLeft 和 helperRight )。 然 后 ， 迭 代 访 问 helper 数 组 ， 将 左右 
两 半 中 较 小 的 元 素 ， 复 制 到 目标 数组 中 。 最 后 ， 再 将 余下 所 有 元 素 复制 回 目标 数组 。 










































































1 void mergesort(int[] array, int low, int high) { 

2 if (low < high) { 

3 int middle = (low + high) / 2; 

4 mergesort(array，low，middle); // 排序 左 半 部 分 

5 mergesort(array，middle + 1，high); // 排序 右 半 部 分 
6 merge(array，low，middle,，high); // 归并 

7 
8 


} 
9 


16 void merge(int[] array, int low, int middle, int high) { 
1 int[] helper = new int[array.length]; 


13 /* 将 数组 左右 两 半 找 贝 到 helper 数 组 中 */ 
14 for (int i = low; i <= high; i++) { 
15 helper[i] = array[i]; 


18 int helperLeft = low; 
19 int helperRight = middle + 1; 
26 int current = low; 


22 /* 选 代 访 问 helper 数 组 。 比 较 左 、 右 两 半 的 元 素 ， 
23 * 并 将 较 小 的 元 素 复 制 到 原先 的 数组 中 。 


24 */ 

25 while (helperLeft <= middle && helperRight <= high) { 
26 if (helper[helperLeft] <= helper[helperRight]) { 
27 array[current] = helper[helperLeft]; 

28 helperLeft++; 

29 } else { // 如 果 右 边 的 元 素 小 于 左边 的 元 素 

36 array[current] = helper[helperRight]; 

31 helperRight++; 

32 } 

33 current++; 

34 } 

35 

36 /* 将 数组 左 半 部 分 剩余 元 素 

37 * 复制 到 目标 数组 中 */ 


38 int remaining = middle - helperLeft; 
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39 for (int i = 6; i <= remaining; i++) { 

46 array[current + i] = helper[helperLeft + i]; 
41 } 

42 } 

43 public static void mergesort(int[] array) { 

44 int[] helper = new int[array.length]; 

45 mergesort(array, helper, 0, array.length - 1); 
46 } 




















你 可 能 会 发 现 , 上 述 代码 只 是 将 helper 数 组 左 半 部 分 剩余 元 素 , 复制 回 目标 数组 中 。 为 什么 
不 复制 右 半 部 分 的 呢 ? 那 是 因为 这 部 分 元 素 早 已 在 目标 数组 中 ， 无 需 复 制 。 

下 面 以 数组 [1，4，5 || 2，8，9] (符号 “| ”表示 分 界 点 ) 为 例 进行 说 明 。 在 合并 左右 两 
部 分 的 元 素 之 前 ，helper 数 组 与 目标 数组 末尾 都 是 [8，9]。 将 4 个 元 素 (1、4、5 和 2 ) 复制 到 目 
标 数组 时 ，[8，9] 仍 在 原 处 。 所 以 ， 也 就 不 需要 复制 这 两 个 元 素 。 

@ 快速 排序 | 执行 时 间 : 平均 情况 为 O(n log(n))， 最 差 情 况 为 O(n”)， 存 储 空 间 : O(log(n)) 

快速 排序 是 随机 挑选 一 个 元 素 ， 对 数组 进行 分 割 ， 以 将 所 有 比 它 小 的 元 素 排 在 前 面 ， 比 它 大 
的 元 素 则 排 在 后 面 。 这 里 的 分 割 经 由 一 系列 元 素 交 换 的 动作 完成 ( 见 下 文 )。 

如 果 我 们 根据 某 元 素 再 对 数组 ( 及 其 子 数组 ) 进行 分 割 ， 并 反复 执行 ,最 后 数组 就 会 变 成 有 
序 的 。 然 而 ， 因 为 无 法 确保 分 割 元 素 就 是 数组 的 中 位 数 (或 接近 中 位 数 )， 快 速 排序 的 效率 可 能 
会 非常 低下 ， 这 也 是 为 什么 最 差 情况 时 间 复 杂 度 为 O(n ) 的 原因 。 



































1 void quickSort(int arr[], int left, int right) { 
2 int index = partition(arr, left, right); 

3 if (left < index - 1) { // 排序 左 半 部 分 

4 quickSort(arr, left, index - 1); 

5 } 

6 if (index < right) { // 排序 右 半 部 分 

7 quickSort(arr, index, right); 

8 } 

9 } 

16 


11 int partition(int arr[], int left, int right) { 
12 int pivot = arr[(left + right) / 2]; // 挑 出 一 个 基准 点 
13 while (left <= right) { 





14 // 找 出 左边 中 应 被 放 到 右边 的 元 素 

15 while (arr[left] < pivot) left++; 
16 

17 // 找 出 右边 中 应 被 放 到 左边 的 元 素 

18 while (arr[right] > pivot) right--; 
19 

26 // 交换 元 素 ， 同 时 调整 左右 索引 值 

21 if (left <= right) { 

22 swap(arr，left，right); // 交换 元 素 
23 left++; 

24 right--; 

25 } 

26 

27 return left; 

28 } 

29 


图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 


76 第 8 章 面试 考题 





@ 基数 排序 | 执行 时 间 : O(fkn) ( 见 下 文 ) 

基数 排序 是 个 整数 (或 其 他 一 些 数据 类 型 ) 排序 算法 ， 充 分 利用 整数 的 位 数 有 限 这 一 事实 。 
使 用 基数 排序 时 ,我 们 会 迭代 访问 数字 的 每 一 位 ， 按 各 个 位 对 这 些 数字 分 组 。 比 如 说 , 假设 有 一 
个 整数 数组 , 我 们 可 以 先 按 个 位 对 这 些 数字 进行 分 组 , 于是, 个 位 为 0 的 数字 就 会 分 在 同一 组 里 。 
然后 ,， 再 按 十 位 进行 分 组 ， 如 此 反复 执行 同样 的 过 程 ， 逐 级 按 更 高 位 进行 排序 ， 直 到 最 后 整个 数 
组 变 为 有 序数 组 。 

其 他 比较 算法 的 平均 情况 执行 时 间 不 会 优 于 O(n log(n))， 相 比 之 下 ， 基 数 排序 的 执行 时 间 为 
O(1m)， 其 中 为 元 素 个 数 ，K 为 数字 的 位 数 。 

2. 查找 算法 

提 到 查找 算法 时 ,我 们 一 般 都 会 想到 二 分 查找 法 。 这 个 算法 的 确 非常 有 用 ,值得 研习 。 在 二 
分 查找 中 ， 要 在 有 序数 组 里 查找 元 素 x， 我 们 会 先 取 数组 中 间 元 素 与 x 作 比 较 。 若 x 小 于 中 间 元 素 ， 
则 搜索 数组 的 左 半 部 。 若 x 大 于 中 间 元 素 ， 则 搜索 数组 的 右 半 部 。 然 后 ， 重 复 这 个 过 程 ， 将 左 半 
部 和 右 半 部 视 作 子 数组 继续 搜索 。 我 们 再 次 取 这 个 子 数 组 的 中 间 元 素 与 x 作 比较 ， 然 后 搜索 左 半 
部 或 右 半 部 。 我 们 会 重复 这 一 过 程 ， 直 至 找到 x 或 子 数组 大 小 为 0。 

概念 上 似乎 很 简单 ,但 要 真正 掌握 全 部 细节 ， 却 比 你 想象 的 还 要 困难 。 研 读 以 下 代码 时 ,请 
注意 哪里 要 加 1、 哪 里 要 减 1。 


























1 int binarySearch(int[] a, int x) { 
2 int low = @; 

3 int high = a.length - 1; 

4 int mid; 

5 

6 while (low <= high) { 

7 mid = (low + high) / 2; 
8 if (a[mid] < x) { 

9 low = mid + 1; 

16 } else if (a[mid] > x) { 
11 high = mid - 1; 

12 } else { 

13 return mid; 

14 } 

15 

16 return -1; // 错误 

17 } 

18 


19 int binarySearchRecursive(int[] a, int x, int low, int high) { 
26 if (low > high) return -1; // 错误 


22 int mid = (low + high) / 2; 
23 if (a[mid] < x) { 


24 return binarySearchRecursive(a, x, mid + 1, high); 
25 } else if (a[mid] > x) { 

26 return binarySearchRecursive(a, x, low, mid - 1); 
27 } else { 

28 return mid; 

29 } 

36 } 
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除了 二 分 查找 法 ,还 有 很 多 种 查找 数据 结构 的 方法 ， 总之, 我 们 不 要 拘泥 于 二 分 查找 法 。 比 
如 说 ， 你 可 以 利用 二 又 树 或 使 用 散 列表 来 查找 某 结 点 。 尽 情 开 拓 思 路 吧 























面试 题目 





11.1 给 定 两 个 排序 后 的 数组 A 和 B， 其 中 A 的 末端 有 足够 的 缓冲 空 容纳 B。 编 写 一 个 方法 , 将 B 合 
并 入 A 并 排序 。( 第 255 页 ) 

11.2 编写 一 个 方法 ， 对 字符 串 数 组 进行 排序 ， 将 所 有 变 位 词 排 在 相 邻 的 位 置 。( 第 256 页 ) 

11.3 给 定 一 个 排序 后 的 数组 ,包含 x 个 整数 , 但 这 个 数组 已 被 旋转 过 很 多 次 ,次 数 不 详 。 请 编写 
代码 找 出 数组 中 的 某 个 元 素 。 可 以 假定 数组 元 素 原 先是 按 从 小 到 大 的 顺序 排列 的 。 
示例 
输入 : 在 数组 (15，16，19，26，25，1，3，4，5，7，16，14} 中 找 出 5。 
输出 : 8 (元 素 5 在 该 数组 中 的 索引 ) ( 第 257 页 ) 

11.4 设想 你 有 个 20GB 的 文件 ,每 一 行 一 个 字符 串 。 请 说 明 将 如 何 对 这 个 文件 进行 排序 。( 第 258 


















































页 ) 

11.5 有 个 排序 后 的 字符 串 数 组 ， 其 中 散布 着 一 些 空 字符 串 ， 编 写 一 个 方法 ， 找 出 给 定 字 符 串 的 
位 置 。 
示例 


输入 : 在 字符 串 数组 {“at”，“”，“，“，“ball”， 2， 人 “car” “dad”， 2， 
“»} 中 ， 查 找 “ball”。 
输出 : 4 (第 259 页 ) 
11.6 给 定 MxN 和 矩阵 ， 每 一 行 、 每 一 列 都 按 升序 排列 ， 请 编写 代码 找 出 某 元 素 。( 第 260 页 ) 
11.7 有 个 马戏 团 正 在 设计 县 罗汉 的 表演 节目 ， 一 个 人 要 站 在 另 一 人 的 肩膀 上 。 出 于 实际 和 美观 
的 考虑 ， 在 上 面 的 人 要 比 下 面 的 人 矮 一 点 、 轻 一 点 。 已 知 马戏 团 每 个 人 的 高 度 和 重量 ， 请 
编写 代码 计算 受 罗 汉 最 多 能 受 几 个 人 。 
示例 
输入 (ht，wt): (65，166) (786，156) (56，96) (75，196) (66，95) (68，116) 
输出 : 从 上 往 下 数 , 到 罗 汉 最 多 能 攻 6 层 : (56, 98) (66,95) (65,168) (68,116) (76,156) 
(75,198) (第 265 页 ) 
假设 你 正在 读 取 一 串 整 数 。 每 隔 一 段 时 间 ， 你 希望 能 找 出 数字 x 的 秩 ( 小 于 或 等 于 x 的 值 的 
数目 )。 请 实现 数据 结构 和 算法 支持 这 些 操作 ， 也 就 是 说 ， 实 现 track(int x) 方 法 ， 每 读 
入 一 个 数字 都 会 调用 该 方法 ; 以 及 getRankofNumber(int x) 方 法 ， 返回 值 为 小 于 或 等 于 x 
的 元 素 个 数 ( 不 包括 x 本 身 )。 
示例 
数据 流 为 ( 按 出 现 的 先后 顺序 ) : 5，1，4，4，5，9，7，13，3 
getRankOfNumber(1) = 6 
getRankOfNumber(3) = 1 























11. 
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getRankOfNumber(4) = 3 (第 267 页 ) 


参考 问题 : 数组 与 字符 事 ( 衣 .3 ) ; 递归 ( 反 .3 ); 中 等 难题 (区 7.6、 衣 7.12 ) ; 高 难度 题 (#8.5 ) 。 
8.12 测试 


在 念 电 着 “我 又 不 是 测试 员 ” 准 备 跳 过 本 章 之 前 ， 请 你 三 思 。 对 于 软件 工程 师 来 说 ， 测 试 是 
项 很 重要 的 工作 ， 因 此 , 在 面试 中 你 很 可 能 会 碰 到 测试 问题 。 当 然 ， 如 果 你 刚好 要 应 聘 测试 职位 
(或 软件 测试 工程 师 )， 那 就 更 应 该 好 好 研读 本 章 。 

测试 问题 一 般 分 为 以 下 四 类 : (1) 测试 现实 生活 中 的 事物 ( 比如 一 支 笔 ); (2) 测试 一 套 软 
件 ; (3) 编写 代码 测试 一 个 函数 ; (4) 调试 解决 已 知 问题 。 针 对 每 一 类 题 型 ， 我 们 都 会 给 出 相应 
的 解法 。 

请 记 住 , 处 理 这 四 类 问题 时 ,， 切 勿 假设 使 用 者 会 好 好 地 正常 操作 。 请 做 好 应 对 用 户 误 用 乱用 
软件 的 准备 。 

1. 面试 官 想 考 察 什么 

表面 上 看 ， 测 试问 题 主要 考察 你 能 否 想 到 周全 完备 的 测试 用 例 。 这 在 某 种 程度 上 也 是 对 的 ， 
求职 者 确实 需要 想 出 一 系列 合理 的 测试 用 例 。 

但 除 此 之 外 ， 面 试 官 还 想 考 察 以 下 几 个 方面 。 

口 全 局 观 : 你 是 否 真 的 了 解 软件 是 怎么 回 事 ? 你 能 否 正确 区 分 测试 用 例 的 优先 顺序 ? 比如 

说 ， 假 设 你 被 问 到 该 如 何 测试 像 亚马逊 这 样 的 电子 商务 系统 。 若 能 确保 产品 图 片 显 示 位 
置 正确 ， 当 然 了 不错， 但 最 重要 的 还 是 确保 支付 流程 万 无 一 失 ， 货 品 能 顺利 地 进入 发 货 
流程 ， 并 且 顾 客 绝对 不 能 被 重复 扣 款 。 

口 懂 整 合 : 你 是 否 了 解 软件 的 工作 原理 ?该 如 何 将 它们 整合 成 更 大 的 软件 生态 系统 ? 假设 

要 测试 谷歌 电子 表格 ( Spreadsheets )， 你 自然 会 想到 测试 文档 的 打开 、 存 储 及 编辑 功能 。 
但 是 ， 实 际 上 ， 和 谷歌 电子 表格 也 是 大 型 软件 生态 系统 的 重要 组 成 部 分 之 一 。 所 以 ， 你 还 
需 将 它 与 Gmail、 各 种 插件 和 其 他 模块 整合 在 一 起 进行 测试 。 

口 会 组 织 : 你 能 否 有 条 有 理 地 处 理 问 题 ? 还 是 处 理 问题 时 毫 无 条 理 ? 被 要 求 提出 照相 机 的 
测试 用 例 时 ， 有 些 求职 者 只 会 一 股 脑 儿 倒 出 一 些 杂 乱 无 章 的 想法 。 而 优秀 的 求职 者 却 能 
将 测试 功能 分 为 几 类 ， 比 如 拍照 、 照 片 管理 、 设 置 ， 等 等 。 在 创建 测试 用 例 时 ， 这 种 结 
构 化 处 理 方法 还 有 助 于 你 将 工作 做 得 更 周全 。 

口 可 操作 : 你 制定 的 测试 计划 是 否 合理 ， 行 之 有 效 ? 比如 ， 如 有 用 户 报告 ， 软 件 会 在 打开 
某 张 图 片 时 崩溃 ， 而 你 却 只 是 要 求 他 们 重新 安 闭 软 件 ， 这 显然 太 不 实际 了 。 你 的 测试 计 
划 必 须 切实 可 行 ， 便 于 公司 操作 落实 。 

倘若 能 在 面试 中 充分 展现 这 些 方面 ,那么 , 你 无 疑 就 是 任何 测试 团队 所 梦 裕 以 求 的 重要 一 员 。 

2. 测试 现实 生活 中 的 事物 

问 及 该 如 何 测 试 一 支 笔 时 ， 有 些 求 职 者 会 感到 莫名 惊 话 。 毕 竟 ,， 你 要 测试 的 不 是 软件 吗 ? 没 
错 ， 但 这 些 关 于 “现实 生活 ”的 问题 其 实 很 常见 。 我 们 先 来 看 看 下 面 这 个 例子 吧 ! 
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比如 有 这 么 一 个 问题 : 如 何 测试 一 枚 回形针 ? 

@ 步骤 1: 使 用 者 是 哪些 人 ? 做 什么 用 ? 

你 需要 跟 面 试 官 讨论 一 下 ， 谁 会 使 用 这 个 产品 ， 做 什么 用 。 回 答 可 能 出 乎 你 的 意料 ， 比 如 ， 
回答 可 能 是 “老师 ， 把 纸张 来 在 一 起 ”或 “艺术 家 ， 为 了 弯 成 动物 的 造型 *。 又 或 者 ， 两 者 丝 要 
考虑 。 这 个 问题 的 答案 ， 将 影响 你 怎么 处 理 后 续 问 题 。 

@ 步骤 2: 有 哪些 用 例 ? 

列 出 回形针 的 一 系列 用 例 , 这 对 解决 问题 很 有 帮助 。 在 这 个 例子 中 , 用例 可 能 是 , 将 纸张 固 
定 在 一 起 ， 且 不 得 破坏 纸张 。 

若是 其 他 问题 ,可 能 会 有 多 个 用 例 。 比 如 ， 某 产品 要 能 够 发 送 和 接收 内 容 , 或 擦 写 和 删除 功 
6， 等 等 。 

@ 步骤 3: 有 哪些 使 用 限制 ? 

使 用 限制 可 能 是 ， 回 形 针 一 次 可 以 夹 最 多 30 张 纸 ， 且 不 会 造成 永久 性 损害 ( 比如 弯 掉 )， 另 
外 ， 可 以 夹 30 到 50 张 纸 时 ， 只 不 过 会 发 生 轻微 变形 。 

同时 , 使 用 限制 也 要 考虑 环境 因素 。 比 如 , 回形针 可 和 否 在 非常 温暖 的 环境 下 ( 33 ~ 43 摄 氏 度 ) 
使 用 ? 在 极 寒 环 境 下 呢 ? 

@ 步骤 4: 压力 与 失效 情况 下 的 状态 如 何 ? 

没有 产品 是 万 无 一 失 的 ， 所 以 ， 在 测试 中 ， 还 必须 分 析 失 效 情况 。 跟 面试 官 探 讨 时 ， 最 好 问 
一 下 在 什么 情况 下 产品 失效 是 可 接受 的 ( 甚至 是 必要 的 )， 以 及 什么 样 才 算 是 失效 。 

举 个 例子 ， 要 你 测试 一 台 洗衣 机 ， 你 可 能 会 认为 洗衣 机 至 少 要 能 洗 30 件 IT 恤衫 或 裤子 。 一 次 
放 进 30 到 45 件 衣服 ， 可 能 会 导致 轻微 失效 ， 因 为 衣物 洗 得 不 够 干净 。 若 超过 45 件 衣物 ， 出 现 极端 
失效 或 许可 以 接受 。 不 过 , 这 里 所 谓 的 极端 失效 ,应 该 是 指 洗衣 机 根本 不 该 进 水 ， 绝 对 不 应 该 让 
水 溢出 来 或 引发 火灾 。 

@ 步骤 5: 如 何 执行 测试 ? 

有 些 情况 下 ， 讨 论 执 行 测试 的 细节 可 能 很 重要 。 比 如 ， 若 要 确保 一 把 椅子 能 正常 使 用 5 年 ， 
你 恐怕 不 会 把 它 放 在 家 里 ， 等 上 5 年 再 来 看 结果 。 相 反 ， 你 需要 定义 何谓 “正常 ”使 用 情况 〈 每 
年 会 在 椅子 上 坐 多 少 次 ? 扶手 呢 ? )。 然 后 ， 除 了 做 一 些 手动 测试 ， 你 可 能 还 会 想到 找 台 机 器 ， 
自动 执行 某 些 功 能 测试 。 

3. 测试 一 套 软件 

测试 软件 与 测试 现实 生活 的 事物 非常 相似 。 主 要 差异 在 于 , 软件 测试 往往 更 强调 执行 测试 的 
细节 。 

请 注意 ， 软 件 测试 主要 有 如 下 两 个 方面 。 

口 手动 测试 与 自动 化 测试 : 理想 情况 下 ， 我 们 当然 希望 能 够 自动 化 所 有 的 测试 工作 ， 不 过 

这 不 太 现实 。 有 些 东 西 还 是 手动 测试 来 的 更 好 ， 因 为 某 些 功 能 对 计算 机 而 言 过 于 定性 ， 
计算 机 很 难 有 效 地 检查 ( 比如 ， 内 容 带 有 淫秽 色情 成 分 )。 此 外 ， 计 算 机 只 能 机 械 地 识别 
明确 告知 过 的 情况 ， 而 人 类 就 不 一 样 了 ， 通 过 观察 可 能 发 现 圳 待 验证 的 新 问题。 因此 ， 
在 测试 过 程 中 ， 无 论 是 人 工 还 是 计算 机 ， 两 者 都 不 可 或 缺 。 
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口 黑 盒 测 试 与 白 盒 测试 : 两 者 的 区 别 反映 了 我 们 对 软件 内 部 机 制 的 掌控 程度 。 在 黑 盒 测 试 
中 ， 我 们 只 关心 软件 的 表象 ， 并 且 仅 测试 其 功能 。 而 在 白 盒 测 试 中 ， 我 们 会 了 解 程序 的 
内 部 机 制 ， 还 可 以 分 别 对 每 一 个 函数 单独 进行 测试 。 我 们 也 可 以 自动 执行 部 分 黑 盒 测试 ， 
只 不 过 难度 要 大 得 多 。 
下 面 介绍 一 种 测试 方法 ， 并 从 头 到 尾 细 述 一 遍 。 
@ 步骤 1: 要 做 黑 盒 测试 还 是 白 盒 测试 ? 
尽管 通常 我 们 会 拖 到 测试 后 期 才 考 虑 这 个 问题 , 但 我 喜欢 早点 做 出 选择 。 不妨 跟 面 试 官 确认 
一 下 ， 要 做 黑 盒 测试 还 是 白 盒 测 试 一 一 或 是 两 者 都 要 。 
@ 步骤 2: 使 用 者 是 哪些 人 ? 做 什么 用 ? 
一 般 来 说 ， 软 件 都 会 有 一 个 或 多 个 目标 用 户 ， 设 计 各 个 功能 时 ， 就 会 考虑 用 户 需求 。 比 如 ， 
若 要 你 测试 一 款 家 长 用 来 监控 网 页 浏览 器 的 软件 , 那么 , 你 的 目标 用 户 既 包括 家 长 ( 实施 监控 过 
滤 哪 些 网 站 )， 又 包括 孩子 ( 有些 网 站 被 过 滤 了 )。 用 户 也 可 能 包括 “访客 ”( 也 就 是 既 不 实施 也 
不 受 监 控 的 使 用 者 )。 
@ 步骤 3: 有 哪些 用 例 ? 
在 监控 过 滤 软 件 中 , 家 长 的 用 例 包括 安装 软件 、 更 新 过 滤 网 站 清单 、 移 除 过 滤 网 站 ， 以 及 供 
他 们 自己 使 用 的 不 受 限 制 的 网 络 。 对 孩子 而 言 ， 用 例 包括 访问 合法 内 容 及 “非法 ”内 容 。 
切记 ， 不 可 和 凭空 想象 来 决定 各 种 用 例 ， 应 该 与 面试 官 交流 讨论 后 确定 。 
@ 步骤 4: 有 哪些 使 用 限制 ? 
大 致 定义 好 用 例 后 ， 我 们 还 需 找 出 确切 的 意思 。“ 网 络 被 过 滤 屏 蔽 ”具体 指 什 么 ”只 过 滤 屏 
珊 “ 非 法 ”网 页 还 是 屏蔽 整个 网 站 ?是 否 要 求 该 软件 具备 “学 习 ” 能 力 ， 从 而 识别 不 良 内 容 ， 抑 
或 内 是 根据 白 名 单 或 黑 名 单 进行 过 滤 ? 若 要 求 具备 学 习 能 力 并 自动 识别 不 良 内 容 , 允许 多 大 的 误 
报 漏 报 率 ? 
@ 步骤 5: 压力 条 件 和 失效 条 件 为 何 ? 
软件 的 失效 是 不 可 避免 的 , 那么 ， 软件 失效 应 该 是 什么 样 的 ?显然 ,就 算 软 件 失效 了 ,也 不 
能 导致 计算 机 宕 机 。 在 本 例 中 ,失效 可 能 是 软件 未 能 屏蔽 本 该 屏蔽 的 网 站 , 或 是 屏蔽 本 来 允许 访 
问 的 网 站 。 对 于 后 一 种 情况 ， 你 或 许 应 该 与 面试 官 讨论 一 下 ， 是 不 是 要 让 家 长 输入 密码 ， 允 许 访 
问 该 网 站 。 
@ 步骤 6: 有 哪些 测试 用 例 ? 如 何 执行 测试 ? 
这 里 才 是 手动 测试 和 自动 测试 以 及 黑 盒 测试 和 白 盒 测试 真正 显示 出 差异 的 地 方 。 
在 步骤 3 和 步 又 4 中 ,我们 初步 拟定 了 软件 的 用 例 , 这 里 会 进一步 加 以 定义 ,并 讨论 该 如 何 执 
行 测 试 。 具 体 需要 测试 哪些 情况 ?” 其 中 哪些 步 又 可 以 自动 化 ? 哪些 又 需要 人 工 介入 ? 
请 记 住 ， 在 有 些 测试 中 ， 虽 然 自动 化 可 以 助 你 一 臂 之 力 ， 但 它 也 有 着 重大 缺陷 。 一 般 来 说 ， 
在 测试 过 程 中 ， 手 动 测试 还 是 少不了 的 。 
对 着 上 面 的 清单 一 步 步 解决 问题 时 , 请 不 要 想到 什么 就 草率 吐露 。 这 会 显得 很 没有 条 理 , 而 
且 你 肯定 会 遗漏 重要 环节 。 相 反 , 你 应 该 组 织 好 自己 的 思路 , 先 将 测试 工作 分 割 为 几 个 主要 模块 ， 
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然后 逐一 展开 分 析 。 这 样 , 不 仅 可 以 给 出 一 份 更 完整 的 测试 用 例 清单 , 而 且 也 显得 你 做 事 有 层次 、 

有 条 理 。 

4. 测试 一 个 函数 

基本 上 , 测试 函数 是 测试 中 最 简单 的 一 种 , 与 面试 官 的 交流 相对 也 会 比较 简短 、 清 晰 , 因为 ， 

测试 一 个 函数 通常 不 外 乎 就 是 验证 输入 与 输出 。 

话说 回来 , 千 万 不 要 忽视 与 面试 官 交 流 的 重要 性 。 对 于 任意 可 能 , 特别 是 如 何 处 理 特定 情况 ， 

你 都 应 该 深究 到 底 。 

假设 要 你 编写 代码 测试 对 整数 数组 排序 的 函数 sort(int[] array) ,可 参考 下 面 的 解决 步 又 。 

@ 步骤 1: 定义 测试 用 例 

一 般 来 说 ， 你 应 该 考虑 以 下 几 种 测试 用 例 。 

口 正常 情况 : 输入 正常 数组 时 ， 该 函数 是 否 能 生成 正确 的 输出 ? 务必 记得 考虑 其 中 的 潜在 
问题 。 比 如 ， 排 序 通常 涉及 某 种 分 制 处理 ， 应 该 要 合理 的 想 一 想 ， 数 组 元 素 个 数 为 奇数 
时 ， 由 于 无 法 均 分 数组 ， 算 法 可 能 无 法 处 理 。 所 以 ， 测 试用 例 必须 涵盖 元 素 个 数 为 偶数 
与 奇数 的 两 种 数组 。 

口 极端 情况 : 传人 空 数 组 会 出 现 什么 问题 ? 或 传人 一 个 很 小 的 数组 ( 只 有 一 个 元 素 )? 此 

外 ， 传 人 非常 大 数组 又 会 如 何 呢 ? 

口 空 指针 和 “非法 ”输入 : 值得 花 时 间 好 好 考虑 一 番 ， 若 函数 接收 到 非法 输入 ， 该 怎么 处 
理 。 比 如 ， 你 在 测试 生成 第 z 项 斐 波 那 契 数 的 函数 ， 那 么 ， 在 测试 用 例 中 ， 自 然 要 考虑 m 
为 负数 的 情况 。 

口 奇怪 的 输入 : 第 四 种 有 可 能 出 现 的 情况 : 奇怪 的 输入 。 传 人 一 个 有 序数 组 会 怎么 样 ? 或 

者 ， 传 人 一 个 反 向 排序 的 数组 呢 ? 

只 有 充分 了 解 函 数 功 能 , 才能 想到 这 些 测试 用 例 。 如 果 你 对 各 种 限制 条 件 不 是 很 清楚 ,最 好 

先 向 面试 官 问 个 清楚 。 

@ 步骤 2: 定义 预期 结果 

通常 , 预期 结果 非常 明显 : 正确 的 输出 。 然 而 , 在 某 些 情况 下 , 你 可 能 还 需要 验证 其 他 情况 。 

比如 ,如果 sort 函 数 返回 的 是 一 个 已 排序 的 新 数组 , 那么 , 你 可 能 还 要 验证 一 下 原先 的 数组 是 否 

保持 原样 。 

@ 步 又 3: 编写 测试 代码 

有 了 测试 用 例 ， 并 定义 好 预期 结果 后 ,编写 代码 实现 这 些 测试 用 例 ， 也 就 水 到 渠 成 了 。 代码 

大 致 如 下 : 



















































































1 void testAddThreeSorted() { 

2 MyList list = new MyList(); 

3 list.addThreeSorted(3，1，2); // 按 顺序 添加 3 个 元 素 
4 assertEquals(1ist.getElement(86)，1); 

5 assertEquals(list.getElement(1), 2); 

6 assertEquals(list.getElement(2), 3); 

了 


} 
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5. 调试 与 故障 排除 

测试 问题 的 最 后 一 种 是 ,说 明 你 会 如 何 调试 或 排除 已 知 故 障 。 碰 到 这 种 问题 ,很 多 求职 者 都 
会 支 支吾 再， 处 理 不 当 ， 给 出 诸如 “ 重 装 软件 ”等 不 切实 际 的 答案 。 其 实 ， 就 像 其 他 问题 一 样 ， 
还 是 有 章 可 循 的 ， 也 可 以 有 条 不 率 地 处 理 。 

下 面 通过 一 个 例子 辅助 说 明 ， 假 设 你 是 谷歌 Chrome 浏 览 器 团队 的 一 员 ， 收 到 一 份 bug 报 告 ; 
Chrome 启 动 时 会 月 溃 。 你 会 怎么 处 理 ? 

重新 安装 浏览 器 或 许 就 能 解决 该 用 户 的 问题 , 但 是 , 若 其 他 用 户 碰 到 同样 问题 ， 怎 么 办 ? 你 
的 目标 是 摘 清楚 究竟 出 了 什么 问题 ， 以 便 开 发 人 员 修复 缺陷 。 

@ 步骤 1: 理 清 状况 

首先 ， 你 应 该 多 提问 题 ， 尽 量 了 解 当时 的 情况 : 
口 用 户 碰 到 这 个 问题 有 多 久 了 ? 
口 该 浏览 器 的 版 本 号 ”在 什么 操作 系统 下 运行 ? 
口 该 问题 经 常 发 生 吗 ? 或 者 ， 出 问题 的 频率 有 多 高 ?什么 时 候 会 发 生 ? 
口 有 无 提交 错误 报告 ? 

@ 步骤 2: 分 解 问题 

了 解 了 问题 发 生 时 的 具体 状况 , 接 下 来 , 着手 将 问题 分 解 为 可 测 模块 。 在 这 个 例子 中 ， 可 以 
设想 出 以 下 操作 步骤 。 

(1) 转 到 Windows 的 “开始 ”菜单 。 

(2) 点 击 Chrome 图 标 。 

(3) 浏览 器 启动 。 

(4) 浏览 器 载 入 参数 设置 。 

(5) 浏览 器 发 送 HTTP 请 求 载 入 首页。 

(6) 浏览 器 收 到 HTTP 回 应 。 








了 


















































(7) 浏览 器 解析 网 页 。 
(8) 浏览 絮 显 示 网 页 内 容 。 
在 上 述 过 程 中 的 某 一 点 ,有 地 方 出 错 致 使 浏览 器 骨 演 ,优秀 的 测试 人 员 会 逐一 排查 每 个 步骤 ， 





诊断 定位 问题 所 在 。 

@ 步骤 3: 创建 特定 的 、 可 控 的 测试 

以 上 各 个 测试 模块 都 应 该 有 实际 的 指令 动作 一 一 也 就 是 你 要 求 用 户 执行 的 、 或 是 你 自己 可 以 
做 的 操作 步骤 (从 而 在 你 自己 的 机 器 上 了 予以 重 现 )。 在 真实 世界 中 ， 你 面 对 的 是 一 般 客户 ， 不 可 
能 给 他 们 做 不 到 或 不 愿 做 的 操作 指令 。 


























面试 题目 





12.1 找 出 以 下 代码 中 的 错误 ( 可 能 不 止 一 处 ): 


1 unsigned int i; 
2 for (i = 160; i >= 0; --i) 
3 printf(“%dN\n”，i); (第 269 页 ) 
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12.2 有 个 应 用 程序 一 运行 就 月 溃 ， 现 在 你 拿 到 了 源码 。 在 调试 器 中 运行 10 次 之 后 ， 你 发 现 该 应 
用 每 次 崩溃 的 位 置 都 不 一 样 。 这 个 应 用 只 有 一 个 线程 , 并 且 只 调用 C 标 准 库 函 数 。 究 竟 是 
入 样 的 编程 错误 导致 程序 崩溃 ? 该 如 何 逐 一 测试 每 种 错误 ? ( 第 270 页 ) 

12.3 有 个 国际 象棋 游戏 程序 使 用 了 方法 : boolean canMoveTo(int x，int y), 这 个 方法 是 Piece 
类 的 一 部 分 ， 可 以 判断 某 个 棋子 能 否 移 动 到 位 置 (x, y)。 请 说 明 你 会 如 何 测试 该 方法 。( 第 
271 页 ) 

12.4 不 借助 任何 测试 工具 ， 该 如 何 对 网 页 进行 负载 测试 ? (第 272 页 ) 

12.5 如 何 测试 一 支 笔 (第 272 页 ) 

12.6 在 一 个 分 布 式 银行 系统 中 ， 该 如 何 测试 一 台 ATM 机 ( 自动 柜员 机 ) ? (第 273 页 ) 
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8.13 C 和 C++ 


好 的 面试 官 不 会 要 求 你 用 自己 不 懂 的 语言 来 编写 代码 。 一 般 来 说 ， 如 果 面 试 官 要 求 你 用 C++ 
写 人 代码， 那么， 应 该 是 你 在 简历 上 提 及 了 C++。 要 是 没 能 记 住所 有 API， 也 不 用 担心 ， 大 部 分 面 
试 官 ( 虽 不 是 全 部 ) 并 不 会 那么 在 意 这 一 点 。 不 过 ， 我 们 仍 建议 你 学 会 基本 的 C++ 语法 ， 这 样 才 
能 轻松 应 对 这 些 问题 。 

1. 类 和 继承 

虽然 C++ 的 类 与 其 他 语言 的 类 有 些 特征 相似 ， 不 过 ， 还 是 有 必要 回顾 一 下 相关 部 分 语法 。 

下 面 的 代码 演示 了 怎样 利用 继承 实现 一 个 基本 的 类 。 


#include 《iostreamy> 
using namespace std; 





























#define NAME_SIZE 56 // 定义 一 个 宏 


class Person { 
int id; // 所 有 成 员 默 认为 私有 (private) 
char name[NAME_SIZE]; 


vaOmwP 哺 


16 public: 

11 void aboutMe() { 

12 cout <<“I am a person.”; 
13 } 

14 }; 


16 class Student : public Person { 
17 public: 

18 void aboutMe() { 

19 cout << “I am a student.”; 
26 } 

21 )}; 


23 int main() { 


24 Student * p = new Student(); 
25 p->aboutMe(); // 打印 “I am a student.” 


图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 





84 第 8 章 面试 考题 





26 delete p; // 注意 | 务必 释放 之 前 分 配 的 内 存 
27 return ©; 
28 } 


在 C++ 中 ， 所 有 数据 成 员 和 方法 均 默 认为 私有 (private )， 可 用 关键 字 pub1lic 修 改 其 属性 。 

2. 构造 函数 和 析 构 函数 

对 象 创建 时 , 会 自动 调用 类 的 构造 函数 。 如 果 没 有 定义 构造 函数 ,编译 絮 会 自动 生成 一 个 默 
认 构 造 函 数 ( Default Constructor )。 男 外 ， 我 们 也 可 以 定义 自己 的 构造 函数 。 














1 Person(int a) { 

2 id = a; 

3 } 

这 个 类 的 数据 成 员 也 可 以 这 样 初始 化 : 
1 Person(int a) : id(a) { 
2 i 

3 } 


在 真正 的 对 象 创建 之 前 , 且 在 构造 函数 余下 部 分 代码 调用 前 , 数据 成 员 id 就 会 被 赋值 。 在 常 
量 数据 成 员 赋 值 时 ( 只 能 赋 一 次 值 )， 这 种 写法 特别 适用 。 

析 构 函数 会 在 对 象 删除 时 执行 清理 工作 。 对 象 销毁 时 , 会 自动 调用 析 构 函数 。 我 们 不 会 显 式 
调用 析 构 函数 ， 因 此 它 不 能 带 参 数 。 

1 ~Person() { 

2 delete obj; // 释放 之 前 这 个 类 里 分 配 的 内 存 

在 前 面 的 例子 中 ， 我 们 将 p 定 义 为 student 类 型 指针 变量 : 

1 Student * p = new Student(); 

2 p->aboutMe(); 


像 下 面 这 样 ， 把 p 定 义 为 Person * 又 会 怎么 样 ? 

1 Person * p = new Student(); 

2 p->aboutMe(); 

这 么 改 的 话 ， 执 行 时 会 打印 “I am a person”。 这 是 因为 函数 aboutMe 是 在 编译 期 决定 的 ， 
也 即 所 谓 的 静态 绑 定 ( static binding ) 机 制 。 

若 要 确保 调用 的 是 student 的 aboutMe 隆 数 实 现 ， 可 以 将 Person 类 的 aboutMe 定 义 为 virtual: 



































class Person { 


virtual void aboutMe() { 
cout << “I am a person.”; 


}; 
class Student : public Person { 


public: 


2 
3 
4 
5 } 
6 
7 
8 
9 
16 void aboutMe() { 
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11 cout << “I am a student.”; 

12 } 

13 )}; 

当 我 们 无 法 (或 不 想 ) 实现 父 类 的 某 个 方法 时 ， 虚 函数 也 能 派 上 用 场 。 例 如 ,设想 一 下 , 我 
们 想 让 student 和 Teacher 继 承 自 Person， 以 便 实现 一 个 共同 的 方法 ， 如 addCourse(string s)。 








不 过 ， 对 Person 调 用 addcourse 方 法 没有 多 大 意义 ， 因为 要 看 对 象 到 底 是 student 还 是 Teacher， 
才能 确定 该 调用 哪个 方法 的 具体 实现 。 

在 这 种 情况 下 ， 我 们 可 能 想 将 Person 类 的 addCourse 定 义 为 虚 图 数 ， 至 于 函数 实现 则 留 
于 类 。 


1 class Person { 

2 int id; // 所 有 成 员 默 认为 私有 

3 char name[NAME_SIZE] ; 

4 public: 

5 virtual void aboutMe() { 

6 cout <<“I am a person.” << endl; 
7 } 

8 virtual bool addCourse(string s) = 
9 ); 

16 

11 class Student : public Person { 

12 public: 

13 void aboutMe() { 

14 cout << “I am a student.” << endl; 
15 } 

16 

17 bool addCourse(string s) { 

18 cout << “Added course “ << s << “to student.” << endl; 
19 return true; 

26 } 

21 ); 

22 


23 int main() { 

24 Person * p = new Student(); 

25 p->aboutMe(); // 打印 “I am a student.” 
26 p->addCourse(“History”); 

27 delete p; 

ee. 


意 ， 将 addCourse 定 义 为 纯 虚 函数 ，Person 就 成 了 一 个 抽象 类 ， 不 能 实例 化 。 
@ 虚 析 构 函 数 
有 了 虚 函 数 , 很 自然 地 就 会 出 现 虚 析 构 函数 的 概念 。 假 设 我 们 想 要 实现 Person 和 student 的 
析 构 函数 。 不 假 思索 的 话 ， 可 能 会 写 出 类 似 如 下 的 代码 : el 





1 class Person { 

2 public: 

3 ~Person() { 

4 cout «<< “Deleting a person.” << endl; 
5 } 

6 )}; 
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yl 

8 class Student : public Person { 

9 public: 

16 ~Student() { 

11 cout << “Deleting a student.” << endl; 
12 } 

13 ); 

14 


15 int main() { 

16 Person * p = new Student(); 

17 delete p; // 打印 “Deleting a person.” 
18 } 


跟 之 前 的 例子 一 样 ， 由 于 指针 p 指 向 Person， 对 象 销毁 时 自然 会 调用 Person 类 的 析 构 函数 。 
这 样 就 会 有 问题 ， 因 为 student 对 象 的 内 存 可 能 得 不 到 释放 。 
要 解决 这 个 问题 ， 只 需 将 Person 的 析 构 函数 定义 为 虚 析 构 函 数 。 











class Person { 

public: 

virtual ~Person() { 

cout << “Deleting a person.” <x< endl; 


}; 


class Student : public Person { 
public: 
16 ~Student() { 
11 cout << “Deleting a student.” << endl; 
12 } 
13 }); 
14 
15 int main() { 
16 Person * p = new Student(); 
17 delete p; 
18 } 


编译 执行 上 面 的 代码 ， 打 印 输 出 如 下 : 


Deleting a student. 
Deleting a person. 


4. 默认 值 
如 下 所 示 ， 函 数 可 以 指定 默认 值 。 注 意 , 所 有 默认 参数 必须 放 在 函数 声明 的 右边 ,因为 没有 
其 他 途径 来 指定 参数 是 怎么 排列 的 。 


1 
2 
3 
4 
5 } 
6 
7 
8 
9 








3 























1 int func(int a, int b = 3) { 
之 x = a; 

3 y= b; 

4 return a + b; 

5 3} 

6 

7 w= func(4); 

8 z= func(4, 5); 
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5. 操作 符 重 载 

有 了 操作 符 重 载 (operator overloading )， 原 本 不 支持 + 等 操作 符 的 对 象 ， 就 可 以 用 上 这 些 操 
作 符 。 举 个 例子 ， 要 想 把 两 个 书架 ( BookShelf ) 并 作 一 个 ， 我 们 可 以 这 样 重 载 + 操作 符 : 

1 BookShelf BookShelf::operator+(BookShelf &other) { ... } 

6. 指针 和 引用 

间 针 存放 有 变量 的 地 址 ， 可 直接 作用 于 变量 的 所 有 操作 ,都 可 以 作用 在 指针 上 ， 比 如 访问 和 
修改 变量 。 

两 个 指针 可 以 彼此 相等 ， 修 改 其 中 一 个 指针 指向 的 值 ， 另 一 个 指针 指向 的 值 也 会 随 之 改变 。 
实际 上 ， 这 两 个 指针 指向 同一 地 址 。 











1 int * p = new int; 

2 *p=7; 

3 int *q=p; 

4 *p= 8; 

5 cout << *q; // 打印 8 





注意 ， 指 针 的 大 小 随 计算 机 的 体系 结构 不 同 而 不 同 : 在 32 位 机 器 上 为 32 位 ， 在 64 位 机 器 上 为 64 
位 。 请 说 记 这 一 点 区 别 , 面试 官 常常 会 要 求 求职 者 准确 地 回答 , 某 个 数据 结构 到 底 要 占用 多 少 空间 。 

@ 引用 

引用 是 既 有 对 象 的 另 一 个 名 字 (别名 )， 引 用 本 身 并 不 占用 内 存 。 例 如 

1 inta=5) 

2 int & b = a; 


3 b= 7; 
4 cout <x< a; // 打印 7 


在 上 面 第 2 行 代码 中 ，b 是 a 的 引用 ; 修改 pb，a 也 随 之 改变 。 
创建 引用 时 , 必须 指定 引用 指向 的 内 存 位 置 。 当然, 也 可 以 创建 一 个 独立 的 引用 , 如 下 所 示 : 
1 /* 分 配 内 存 ， 储 存 12， 


2  * 声明 指向 这 块 内 存 的 引用 b */ 
3 int&b = 12; 


跟 指 针 不 同 ， 引 用 不 能 为 空 ， 也 不 能 重新 赋值 ， 指 向 另 一 块 内 存 。 


@ 指针 草 术 运 算 
我 们 经 常会 看 到 开发 人 员 对 指针 执行 加 法 操作 ， 示 例如 下 : 





























int * p = new int[2]; 


1 

2 ple] = @; 

3 p[1] = 1; 

4 p++; 

5 cout << *p; // 输出 1 





执行 p++ 会 跳 过 sizeof(int) 个 字 节 ， 因 此 上 面 的 代码 会 输出 1。 如 果 p 换 作 其 他 类 型 ，p++ 就 
会 跳 过 一 定数 目 ( 等 于 该 数据 结构 的 大 小 ) 的 字 节 。 

7. 模板 

模板 是 一 种 代码 重用 方式 ， 不 同 的 数据 类 型 可 以 套用 同一 个 类 的 代码 。 比 如 说 ， 我 们 可 能 


图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 





88 第 8 章 面试 考题 








列表 类 的 数据 结构 , 希望 可 以 放 进 不 同类 型 的 数据 。 下 面 的 代码 通过 shiftedList 类 实 











1 template <class T> 

2 class ShiftedList { 

3 T* array; 

4 int offset, size; 

5 public: 

6 ShiftedList(int sz) : offset(60), size(sz) { 
7 array = new T[sizel]; 

8 

9 


} 


16 ~ShiftedList() { 


1 delete [] array; 

12 } 

13 

14 void shiftBy(int n) { 

15 offset = (offset + n) % size; 

16 } 

17 

18 T getAt(int i) { 

19 return array[convertIndex(i)]; 
26 } 

21 

22 void setAt(T item, int i) { 

23 array[convertIndex(i)] = item; 
24 } 

25 

26 private: 

27 int convertIndex(int i) { 

28 int index = (i - offset) % size; 
29 while (index < 6) index += size; 
36 Peturn index; 

31 } 

32 ); 

33 

34 int main() { 

35 int size = 4; 

36 ShiftedList<int> * list = new ShiftedList<int>(size); 
37 for (int i = 6;j i < size; i++) { 
38 list->setAt(i, i); 

39 } 


46 cout “< 1List->getAt(9) << end]; 
41 cout “< list->getAt(1) << endl; 
42 list->shiftBy(1); 

43 cout << list->getAt(60) << endl; 
44 cout << list->getAt(1) << endl; 
45 delete list; 

46 } 


现 


\ 


这 一 


和 
TI 


Ek 
求 。 





面试 题目 





13.1 用 C++ 写 个 方法 ， 打 印 输入 文件 的 最 后 K 行 。( 第 274 页 ) 
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13. 


DD 


比较 并 对 比 散 列表 和 STL map。 散 列表 是 怎么 实现 的 ? 如 果 输 入 的 数据 量 不 大 ， 可 以 选用 

哪些 数据 结构 替代 散 列 表 ? (第 275 页 ) 

13.3 C++ 虚 函数 的 工作 原理 是 什么 ? ( 第 275 页 ) 

13.4 深 拷贝 和 浅 拷贝 之 间 有 何 区 别 ? 请 说 明 两 者 的 用 法 。( 第 276 页 ) 

13.5 C 语 言 的 关键 字 volatile 有 何 作 用 ? (第 277 页 ) 

13.6 基 类 的 析 构 函数 为 何 要 声明 为 virtual? (第 278 页 ) 

13.7 编写 方法 ， 传 人 参数 为 指向 Node 结 构 的 指针 ， 返 回 传人 数据 结构 的 完整 拷贝 。 其 中 ，Node 

数据 结构 含有 两 个 指向 其 他 Node 的 指针 。( 第 278 页 ) 

13.8 编写 一 个 智能 指针 类 。 智 能 指针 是 一 种 数据 类 型 ， 一 般 用 模板 实现 ， 模 拟 指针 行为 的 同时 
还 提供 自动 垃圾 回收 机 制 。 它 会 自动 记录 smartPointer<T*> 对 象 的 引用 计数 ,一 旦 T 类 型 对 
象 的 引用 计数 为 零 ， 就 会 释放 该 对 象 。( 第 279 页 ) 

13.9 编写 支持 对 齐 分 配 的 malloc 和 free 函 数 ， 分 配 内 存 时 ，malloc 函 数 返 回 的 地 址 必须 能 被 2 
的 n 次 方 整除 。 
示例 
align_malloc(1666,128) 返 回 的 内 存 地 址 可 被 128 整 除 , 并 指 问 一 块 1666 字 市 大 小 的 内 存 。 
aligned_free() 会 释放 align_malloc 分 配 的 内 存 。( 第 281 页 ) 

13.10 用 C 编 写 一 个 my2DAl1loc 函 数 ， 可 分 配 二 维 数组 。 将 malloc 函 数 的 调用 次 数 降 到 最 少 ， 并 

确保 可 通过 arr[i][j] 访 问 该 内 存 。( 第 282 页 ) 


参 者 问题， 数组 与 字符 串 (#1.2 ) ; 链表 (所 .7) ; 测试 (#2.1 ) ; Java (#14.4 ) ; 线程 与 
锁 (#16.3 ) 。 



























































8.14 Java 


虽然 本 书 到 处 都 是 跟 Java 相 关 的 问题 ， 不 过 ， 本 章 探讨 的 是 Java 及 其 语法 方面 的 问题 。 较 大 
的 公司 通常 不 会 考 这 类 问题 , 这 些 公司 偏重 于 测试 求职 者 的 资质 而 非 知 识 , 也 有 时 间 和 资源 就 特 
定语 言 对 求职 者 进行 培训 。 不 过 ， 千 在 其 他 公司 ， 这 类 琼 手 的 问题 可 能 相当 常见 。 

1. 如 何 处 理 

既然 这 些 问 题 考 的 是 你 知道 不 知道 ,讨论 这 类 问题 的 解法 似乎 有 点 可 笑 。 毕 竞 ， 所 谓 的 解法 
不 就 是 要 知道 正确 管 案 吗 ? 

既是 也 不 是 。 当 然 ， 掌握 这 些 问 题 的 最 佳 途径 就 是 搞 懂 Java 的 里 里 外 外 。 不 过 ， 若 在 处 理 问 
题 时 卡 膏 了 ， 不 妨 试 试 下 面 的 方法 。 

(1) 根据 情况 创建 实例 ， 问 问 自己 该 如 何 推演 。 

(2) 问 问 自己 ， 换 作 其 他 语言 ， 该 怎么 处 理 这 种 情况 。 

(3) 如 果 你 是 语言 设计 者 ， 该 怎么 设计 ? 各 种 设计 选择 都 会 造成 什么 影响 ? 

相 比 不 假 思 索 地 答 出 问题 ， 如 果 你 能 推导 出 答案 ， 同 样 会 给 面试 官 留 下 深刻 的 印象 。 不 要 
试图 蒙混 过 关 。 你 可 以 直接 告诉 面试 官 :“ 我 不 确定 能 否 想起 答案 ， 不 过 让 我 试 试 能 不 能 搞定 
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它 。 假 设 我 们 拿 到 这 段 代 码 …… . 
@ 关键 字 final 
Java 语 言 的 关键 字 final 用 于 变量 、 类 或 方法 时 ,含义 各 不 相同 。 
口 变量 : 一 旦 初始 化 ， 变 量 值 就 不 能 修改 。 
口 方法 : 该 方法 不 能 被 子 类 重 写 ( override )。 
口 类 : 该 类 不 能 派生 子 类 。 
@ 关键 字 finally 
关键 字 finally 和 try/catch 语 句 块 配对 使 用 ， 即 使 有 异常 抛 出 ， 也 能 确保 某 段 代码 一 定 会 
执行 。finally 语 句 块 会 在 try 和 catch 语 句 块 之 后 ， 在 控制 权 交 回 之 前 执行 。 
注意 ， 下 面 这 个 例子 中 该 关键 字 是 怎么 起 作用 的 。 


1 public static String lem() { 
2 System.out.println(“lem”); 





























3 return “return from lem”; 

4 } 

5 

6 public static String foo() { 

7 int x = @; 

8 int y = 5; 

9 try { 

16 System.out.println(“start try”); 
生 和 int b=y/ x; 

42 System.out.println(“end try”); 
13 return “returned from try”; 

14 } catch (Exception ex) { 

15 System.out.println(“catch”); 

16 return lem() + “ | returned from catch”; 
17 } finally { 

18 System.out.println(“finally”); 
19 } 

20 } 


22 public static void bar() { 

23 System.out.println(“start bar”); 
24 String v = foo(); 

25 System.out.println(v); 

26 System.out.println(“end bar”); 
27 } 


29 public static void main(String[] args) { 
36 bar(); 

31 } 

这 段 代码 的 输出 如 下 : 

1 start bar 

2 start try 

3 catch 

4 lem 

5 finally 


6 return from lem | returned from catch 
7 end bar 
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注意 上 述 输出 的 第 3 ~ 5 行 。 整 个 catch 语 句 块 都 会 执行 ( 包括 return 语 句 里 的 函数 调用 )， 
然后 执行 finally 语 句 块 ， 之 后 该 函数 才 真 正 返回 。 

@ finalize 方 法 

在 真正 销毁 对 象 之 前 ， 自 动 垃圾 收集 需 会 调用 finalize() 方 法 。 因 此 ， 一 个 类 可 以 重 写 
Object 类 的 finalize() 方 法 ， 以 便 定 义 在 垃圾 收集 时 的 特定 行为 。 


1 protected void finalize() throws Throwable { 
2 /* 关闭 已 打开 的 文件 ， 释 放 资 源 等 */ 








3 } 

2. 重 载 与 重 写 

重 载 (overloading ) 是 指 两 个 方法 的 名 称 相同 ， 但 参数 类 型 或 个 数 不 同 。 
1 public double computeArea(Circle c) { ... } 

2 public double computeArea(Square s) { ...} 


而 重 写 (overriding ) 是 指 某 个 方法 与 父 类 的 方法 拥有 相同 的 名 称 和 函数 签名 。 


1 public abstract class Shape { 

2 public void printMe() { 

3 System.out.println(“I am a shape.”); 
4 } 

5 public abstract double computeArea( ) ; 
6 } 

yl 

8 public class Circle extends Shape { 

9 private double rad = 5; 

16 public void printMe() { 

11 System.out.println(“I am a circle.”); 
12 } 

13 

14 public double computeArea() { 

25 return rad * rad * 3.15; 

16 } 

17 } 

18 

19 public class Ambiguous extends Shape { 
26 private double area = 10; 

2 public double computeArea() { 

22 return area; 

23 } 

24 } 

25 


26 public class IntroductionOverriding { 
27 public static void main(String[] args) { 





28 Shape[] shapes = new Shape[2]; 

29 Circle circle = new Circle(); 

36 Ambiguous ambiguous = new Ambiguous(); 
31 

32 shapes[6] = circle; 

33 shapes[1] = ambiguous; 

34 
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35 for (Shape s : shapes) { 
36 s.printMe(); 
37 System.out.println(s.computeArea()); 
38 } 
39 } 
46 } 


这 段 代 码 的 输出 如 下 : 


1 
2 
3 


4 


I am a circle. 
78.75 
I am a shape. 
16.6 


由 此 可 见 ，circle 重 写 了 printMe() ， 但 Ambiguous 并 未 重 写 该 方法 。 

3. 集合 框架 

Java 的 集合 框架 〈 collection framework ) 极其 有 用 ， 本 书 许多 章节 都 用 到 了 。 下 面 介绍 几 个 
最 常用 的 。 

ArrayList: ArrayList 是 一 种 可 动态 调整 大 小 的 数组 ， 随 着 元 素 的 插入 ， 数 组 会 适时 扩容 。 


PuUDPp 








ArrayList<String> myArr = new ArrayList<String>(); 
myArr.add(“one”); 

myArr.add(“two”); 
System.out.println(myArr.get(8)); /* 打印 <one> */ 


Vector: Vector 与 ArrayList 非 常 类 似 ， 只 不 过 前 者 是 同步 的 〈synchronized )。 两 者 语法 也 


相差 无 几 。 
1 Vector<String> myVect = new Vector<String>(); 
2 myVect.add(“one”); 
3 myVect.add(“two”); 
4 System.out.println(myVect.get(6) ); 


LinkedList: 这 里 说 的 LinkedList 当 然 是 Java 内 建 的 LinkedList 类 。LinkedList 在 面试 中 
很 少 出 现 ， 不 过 值得 学 习 研 究 ， 因 为 使 用 时 会 引出 一 些 迭 代表 的 语法 。 


NOAAwWoOPp 


HashMap: HashMap 集 合 广泛 用 于 各 种 场合 ， 不 论 是 在 面试 中 ， 还 是 在 实际 开发 中 。 下 面 展 





LinkedList<String> myLinkedList = new LinkedList<String>(); 
myLinkedList.add(“two”); 
myLinkedList.addFirst(“one”); 
Iterator<String> iter = myLinkedList.iterator(); 
while (iter.hasNext()) { 
System.out.println(iter.next()); 


} 











示 了 HashMap 的 部 分 语法 。 


POUDPp 


HashMap<String, String> map = new HashMap<String, String>(); 
map.put(“one”, “uno”); 

map.put(“two”, “dos”); 

System.out.println(map.get(“one”) ); 


由 


面试 之 前 ， 确 保 自己 对 上 述 语 法 了 如 指 掌 。 这 些 语 法 派 得 上 用 场 。 
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面试 题目 








请 注意 ， 本 书 几 乎 所 有 问题 的 解决 方法 都 采用 Java 实 现 ， 因 此 这 里 只 列 了 几 个 问题 。 而 且 ， 

这 些 问 题 主要 涉及 Java 语 言 的 细 枝 未 节 ， 毕 竟 本 书 其 余 章节 中 有 很 多 Java 有 关 的 编程 问题 。 

14.1 从 继承 的 角度 来 看 ， 将 构造 函数 声明 为 私有 会 有 何 作用 ? (第 284 页 ) 

14.2 在 Java 中 ， 阁 在 try-catch-finally 的 try 语 句 块 中 插入 return 语 句 ，finally 语 句 块 是 否 
还 会 执行 ? (第 284 页 ) 

14.3 final、finally 和 finalize 之 间 有 何 差 异 ? (第 285 页 ) 

14.4 C++ 模板 和 Java 泛 型 之 间 有 何不 同 ?” (第 285 页 ) 

14.5 Java 中 的 对 象 反 射 是 什么 ? 它 有 什么 用 ? (第 287 页 ) 

14.6 实现 circularArray 类 ， 支 持 类 似 数组 的 数据 结构 ， 这 些 数据 结构 可 以 高 效 地 进行 旋转 。 
该 类 应 该 使 用 泛 型 ， 并 通过 标准 的 for (0bj o : circularArray) 语 法 支持 迭代 操作 。( 第 
287 页 ) 


参考 问题 ， 数 组 与 字符 串 (#1.4) ; 面向 对 象 设计 (#8.10 ) ; 线程 与 锁 (#16.3 ) 。 


8.15 数据 库 


有 数据 库 经 验 的 求职 者 可 能 会 被 要 求实 现 SQL 查 询 , 或 是 设计 应 用 程序 所 需 的 数据 库 ， 以 确 
认 你 掌握 这 方面 的 知识 。 本 章 将 回顾 一 些 关 键 概念 ， 并 简 述 如 何 解决 这 些 问 题 。 

看 到 这 些 查询 时 ， 对 于 语法 上 的 细微 差异 ， 不 必 太 惊讶 。SQL 的 版 本 和 变 体 很 多 ， 下 面 这 些 
SQL 与 你 之 前 接触 过 的 可 能 稍 有 不 同 。 本 书 的 SQL 示例 已 在 微软 SQL Server 经 过 测试 。 

1. SQL 语法 及 各 类 变 体 

开发 人 员 常 常会 在 SQL 查询 中 使 用 隐 式 连接 (implicitjoin ) 和 显 式 连接 ( explicitjoin )。 两 者 
的 语法 如 下 。 
/* 显 式 连接 */ 
SELECT CourseName, TeacherName 


FROM Courses INNER JOIN Teachers 
ON Courses.TeacherID = Teachers.TeacherID 
























































卢 


/* 隐 式 连接 */ 

SELECT CourseName, TeacherName 

FROM Courses, Teachers 

WHERE Courses.TeacherID = Teachers.TeacherID 


上 面 两 条 语句 的 作用 是 等 价 ， 至 于 选用 哪 条 全 看 个 人 喜好 。 为 保持 前 后 一 致 ， 我们 将 一 直 使 
用 显 式 连 接 。 

2. 非 规范 化 和 规范 化 数据 库 

规范 化 数据 库 的 设计 目标 是 将 元 余 降 到 最 低 ， 而 非 规范 化 数据 库 则 是 为 了 优化 读 取 时 间 。 

在 传统 的 规范 化 数据 库 中 ， 若 有 诸如 Courses 和 Teachers 的 数据 ，Courses 可 能 含有 
TeacherID 列 ， 这 是 指向 Teachers 的 外 键 ( foreign key )。 这么 做 的 好 处 之 一 是 ， 关 于 教师 的 信息 


oNAAUUPUWN 
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( 姓名、 住址 等 ) 在 数据 库 中 只 有 一 份 。 而 缺点 是 大 量 常用 的 查询 需要 执行 开销 很 大 的 连接 操作 。 
反之 , 我 们 可 以 存储 元 余数 据 , 使 数据 库 非 规范 化 。 例 如 , 若 能 预计 到 这 类 查询 会 频繁 执行 ， 
可 以 将 教师 姓名 存 到 courses 表 中 。 非 规范 化 通常 用 于 构建 高 可 扩展 性 系统 。 
3. SQL 语句 
下 面 以 前 面 提 到 的 数据 库 为 例 ， 复 习 一 下 基本 的 SQL 语法 。 这 个 数据 库 的 简单 结构 如 下 ,其 
中 * 表 示 主 键 : 
Courses: CourseID*, CourseName, TeacherID 
Teachers: TeacherID*, TeacherName 


Students: StudentID*, StudentName 
StudentCourses: CourseID*, StudentID* 


根据 上 面 这 些 表 ， 实 现下 列 查 询 。 
@ 查询 1: 学 生 选 课 情 况 
实现 一 个 查询 ， 列 出 所 有 学 生 ， 以 及 每 个 学 生 选 修了 几 门 课程 。 
首先 ， 我们 或 许可 以 试 着 这 么 写 : 
/* 错误 的 代码 */ 
SELECT Students.StudentName, count(*) 
FROM Students INNER JOIN StudentCourses 
ON Students.StudentID = StudentCourses.StudentID 
GROUP BY Students.StudentID 
上 述 查 询 有 以 下 三 个 问题 。 
(我 们 将 一 门 课 都 没 选 的 学 生 排 除 掉 了 ， 因 为 StudentCourses 只 包括 已 经 选课 的 学 生 。 我 
们 可 以 把 INNER JOIN 改 为 LEFT JOIN ( 左 连接 )。 
(2) 即使 改 为 LEFT JOIN,， 上 面 的 查询 还 是 不 太 对 。count(*) 会 返回 一 组 StudentID 里 有 几 项 。 
一 门 课 都 没 选 的 学 生 在 对 应 的 组 里 仍 有 一 项 。 这 里 需要 将 count(*) 改 为 计数 每 个 组 里 CourseID 
的 数量 : count(StudentCourses.CourseID)。 
(3) 上 面 的 查询 已 按 students.StudentID 分 组 ， 但 每 个 组 仍 有 多 个 studentName。 数 据 库 怎 
么 知道 该 返回 哪个 studentName? 当然 ,它们 的 值 可 能 都 一 样 ， 但 数据 库 并 不 知道 这 点 。 这 里 需 
要 运用 聚合 (aggregate ) 函数 ， 比 如 first(Students.StudentName)。 
修正 上 述 问 题 后 ， 得 到 如 下 查询 : 


1 /* 解法 1: 用 另 一 个 查询 包 襄 起 来 */ 

2 SELECT StudentName, Students.StudentID, Cnt 

3 FROM ( 

4 SELECT Students.StudentID, 

5 count(StudentCourses.CourseID) as [Cnt] 
6 FROM Students LEFT JOIN StudentCourses 
7 
8 
9 









































UBUDOP 














ON Students.StudentID = StudentCourses.StudentID 
GROUP BY Students.StudentID 
) T INNER JOIN Students on T.studentID = Students.StudentID 


看 到 这 段 代 码 ， 有 人 可 能 会 问 ， 为 什么 不 直接 在 第 3 行 里 选 出 学 生 姓 名 ， 就 不 需要 第 3 到 第 6 
行 的 另 一 个 查询 了 。 这 么 做 的 话 ， 就 会 得 到 如 下 〈 错误 的 ) 解法 : 
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/* 错误 的 代码 */ 

SELECT StudentName, Students.StudentID, 
count(StudentCourses.CourseID) as [Cnt] 

FROM Students LEFT JOIN StudentCourses 

ON Students.StudentID = StudentCourses.StudentID 

GROUP BY Students.StudentID 


答案 是 我 们 不 能 这 人 么 改 一 一 至 少 不 能 一 五 一 十 照 上 面 那样 改 。 我 们 4 
GROUP BY 子 句 里 的 值 。 
另外 ， 我 们 可 以 使 用 下 面 两 条 语句 之 一 解决 上 面 的 问题 : 


QQ 上 WwW 





的 





选择 聚 


» 
Ea 


- 
Fl 
这 
> 


1 /* 解法 2: 在 GROUP BY 子 句 中 加 入 StudentName +*/ 

2 SELECT StudentName, Students.StudentID, 

3 count(StudentCourses.CourseID) as [Cnt] 

4 FROM Students LEFT JOIN StudentCourses 

5 ON Students.StudentID = StudentCourses.StudentID 

6 GROUP BY Students.StudentID, Students.StudentName 

或 

1 /* 解法 3: 用 聚合 函数 包 训 起 来 */ 

2 SELECT max(StudentName) as [StudentName], Students.StudentID, 
3 count(StudentCourses.CourseID) as [Count] 
4 FROM Students LEFT JOIN StudentCourses 

5 ON Students.StudentID = StudentCourses.StudentID 

6 GROUP BY Students.StudentID 


@ 查询 2: 教师 班级 规模 

实现 一 个 查询 ， 取得 一 份 所 有 教师 的 列表 ， 以 及 每 位 教师 要 教 多 少 学 生 。 如 果 一 位 教师 给 某 
个 学 生 教 授 两 门 课程 ,那么 ,这 个 学 生 就 要 计 人 两 次 。 根 据 教 师 教 授 的 学 生 人 数 , 将 结果 列表 从 
大 到 小 进行 排序 。 

下 面 逐 步 构造 这 个 查询 。 首 先 , 取得 一 份 TeacherID 列 表 , 以 及 有 多 少 学 生 跟 各 个 TeacherID 
有 关联 。 这 跟前 一 个 查询 非常 相似 。 
SELECT TeacherID, count(StudentCourses.CourseID) AS [Number] 
FROM Courses INNER JOIN StudentCourses 


ON Courses.CourseID = StudentCourses.CourseID 
GROUP BY Courses.TeacherID 


青 注意 ， 这 里 的 INNER JOIN 不 会 选取 那些 不 教 课 的 教师 。 我 们 会 在 下 面 的 查询 中 进行 处 理 ， 
将 它 与 包含 所 有 教师 的 列表 相连 接 。 








上 UN 情 


SELECT TeacherName，isnul1(StudentSize.Number，6) 
FROM Teachers LEFT JOIN 
(SELECT TeacherID, count(StudentCourses.CourseID) AS [Number] 
FROM Courses INNER JOIN StudentCourses 
ON Courses.CourseID = StudentCourses.CourseID 
GROUP BY Courses.TeacherID) StudentSize 
ON Teachers.TeacherID = StudentSize.TeacherID 
ORDER BY StudentSize.Number DESC 





1 
之 
3 
4 
5 
6 
7 
8 


请 注意 ， 上 面 的 查询 是 如 何在 SELECT 语句 中 处 理 NULL 值 的 : 将 NULL 值 转换 为 零 。 
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4. 小 型 数据 库 设计 

另外 , 面试 官 或 许 会 让 你 自己 设计 一 个 数据 库 。 下 面 会 逐步 剖析 一 种 设计 方法 。 你 可 能 会 
现 该 方法 与 面向 对 象 设计 方法 存在 相似 之 处 。 

@ 步骤 1: 处 理 不 明确 的 地 方 

不 管 是 有 意 还 是 无 意 , 数据 库 问 题 往往 存在 含糊 不 清 的 地 方 。 开 始 设计 之 前 ， 你 必须 准确 理 
解 自 己 要 设计 什么 。 

设想 一 下 ， 你 被 要 求 设计 一 套 系统 ， 供 公寓 租赁 中 介 使 用 。 你 需要 弄 清 楚 这 家 中 介 有 多 标 
楼 还 是 只 有 一 栋 ， 而 且 还 应 该 跟 面 试 官 讨论 系统 的 通用 性 要 做 到 什么 程度 。 比 如 ， 某 人 租用 同 
一 栋 楼 里 的 两 套 公寓 的 情况 极为 少见 ,但 这 是 和 否 意味 着 你 用 不 着 处 理 这 种 情况 ” 也许 是 ,也许 
不 是 。 有 些 非常 罕见 的 条 件 或 许 最 好 做 变通 处 理 ( 比如 ， 在 数据 库 中 ,重复 存储 承租 人 的 联系 
信息 )。 

@ 步骤 2 : 定义 核心 对 象 

接 下 来 ， 该 来 看 看 系统 的 核心 对 象 了 。 一 般 来 说 ， 每 个 核心 对 象 都 会 转变 为 一 张 表 。 在 这 个 
例子 中 ， 核 心 对 象 可 能 包括 Property (财产 )、Building (大 楼 )、Apartment (公寓 )、Tenant 
(承租 人 ) 和 Manager ( 管理 员 )。 

@ 步骤 3: 分 析 表 之 间 的 关系 

勾勒 出 核心 对 象 后 , 我 们 就 可 以 比较 清晰 地 知道 这 些 表 该 是 什么 样 的 。 这 些 表 之 间 有 何 关联 
呢 ? 它们 的 关系 是 多 对 多 ? 还 是 一 对 多 ? 

若 Building 和 Apartment 有 一 对 多 的 关系 (一 幢 Building 会 有 很 多 Apartment )， 那 么 ， 也 





涉 
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许可 以 表示 如 下 : 
Buildings 
BuildingID BuildingName BuildingAddress 
Apartments 
ApartmentID ApartmentAddress BuildingID 








注意 ，Apartments 表 通过 BuildingID 列 链接 回 Buildings。 
若 人 允许 承租 人 租用 多 套 公 寓 ， 那 么 ， 可 能 就 要 实现 多 对 多 关系 ， 如 下 所 示 : 























Tenants 

TenantID TenantName TenantAddress 
Apartments 

ApartmentID ApartmentAddress BuildingID 
TenantApartments 

TenantID ApartmentID 
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TenantApartments 表 储存 Tenants 和 Apartments 之 间 的 关系 。 


@ 步骤 4: 研究 该 有 什么 操作 动作 


最 后 ,我 们 要 填充 细 方 





。 想 想 常 见 的 操 


作 动 作 ， 弄 清楚 


处 理 租赁 条 款 、 腾 空房 间 、 租 金 付 款 等 。 每 个 动作 都 需要 新 的 表 和 列 。 
5. 大 型 数据 库 设计 
在 设计 大 型 、 可 扩展 的 数据 库 时 ， 上 述 例子 用 到 的 连接 (join ) 通常 都 很 慢 。 因 此 ， 你 必须 




















如 何 存 和 人 和 取 回 相关 数据 。 我 们 还 需 





对 数据 做 非 规范 化 处 理 。 请 仔细 想 想 数据 会 怎么 使 用 一 一 你 可 能 需要 在 多 个 表 中 重复 储存 同一 份 
数据 。 
面试 题目 





问题 1 ~ 3 用 到 以 下 数据 库 模 式 : 


Apartments 


Buildings 


Tenants 





AptID 


int 


BuildingID 


TenantID 


int 





UnitNumber 


varchar 


ComplexID 


TenantName 


varchar 





BuildingID 


int 


BuildingName 


varchar 





Complexes 


Address 


AptTenants 


varchar 


Requests 





ComplexID 


int 


TenantID 


RequestID 


int 





ComplexName 


varchar 


AptID 


Status 


varchar 





AptID 


int 

















Description 








varchar 








注意 每 套 公 寅 可 能 有 多 位 承租 人 , 而 每 位 承租 人 可 能 租 住 多 套 公寓 。 每 套 公 离 只 属于 一 栋 大 

楼 ， 而 每 栋 大 楼 属于 一 个 综合 体 。 

15.1 编写 SQL 查询 ， 列 出 租 住 不 止 一 套 公 寓 的 承租 人 。( 第 290 页 ) 

15.2 编写 SQL 查询 , 列 出 所 有 建筑 物 , 并 取得 状态 为 “Open” 的 申请 数量 ( Requests 表 中 Status 
为 0pen 的 条 目 )。( 第 291 页 ) 

15.3 11 号 建筑 物 正在 进行 大 翻修 。 编 写 SQL 查 询 ， 关 闭 这 栋 建 筑 物 里 所 有 公寓 的 和 人 住 申 请 。( 第 
291 页 ) 

15.4 连接 有 哪些 不 同类 型 ? 请 说 明 这 些 类 型 之 间 的 差异 ， 以 及 为 何在 某 些 情形 下 ， 某 种 连接 会 
比较 好 。( 第 291 页 ) 

15.5 什么 是 反 规 范 化 ?” 请 说 明 优 缺 点 。( 第 292 页 ) 

15.6 有 个 数据 库 ， 里 面 有 公司 (companies )、 人 ( people ) 和 专业 人 员 (professionals， 为 
公司 工作 )， 请 绘制 实体 关系 图 。( 第 293 页 ) 

15.7 给 定 一 个 储存 有 学 生成 绩 的 简单 数据 库 。 设 计 这 个 数据 库 的 大 概 样子 ， 并 编写 SQL 查询 ， 
返回 优等 生 名 单 〈 排 名 前 10% )， 以 平均 分 排序 。( 第 293 页 ) 


参考 问题 : 面向 对 象 设计 (#8.6 ) 。 
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8.16 ”线程 与 锁 


在 微软 、 谷 歌 或 亚马逊 等 公司 的 面试 中 , 求职 者 被 要 求 以 线程 实现 算法 的 情况 并 不 是 很 常见 
( 除非 你 打算 加 入 的 团队 特别 看 重 这 方面 的 技能 ),。 不 过 , 不 管 是 什么 公司 ,面试 官 常常 会 考 你 对 
线程 有 没有 一 定 程度 的 了 解 ， 特 别 是 对 死 锁 的 理解 。 

本 章 将 简要 介绍 这 个 主题 。 

1. Java 线 程 

在 Java 中 , 每 个 线程 的 创建 和 控制 都 是 由 java.lang.Thread 类 的 独特 对 象 实现 的 。 一 个 独立 
的 应 用 运行 时 ， 会 自动 创建 一 个 用 户 线程 ， 执 行 main() 方 法 。 这 个 线程 叫 作 主线 程 。 
在 Java 中 ， 实 现 线程 有 以 下 两 种 方式 : 
口 通过 实现 java.1ang.Runnable 接 口 ; 
口 通过 扩展 java.lang.Thread 类 。 
下 面 介绍 这 两 种 方式 。 
@ 实现 Runnab1le 接 口 
Runnable 接 口 的 结构 非常 简单 : 


1 public interface Runnable { 
2 void run(); 


3 } 

要 用 这 个 接口 创建 和 使 用 线程 ， 步 又 如 下 。 

(1) 创建 一 个 实现 Runnable 接 口 的 类 ， 该 类 的 对 象 是 一 个 Runnable 对 象 。 

(2) 创建 一 个 Thread 类 型 的 对 象 ， 并 将 Runnable 对 象 作为 参数 传人 Thread 构 造 函 数 。 于 是 ， 
这 个 Thread 对 象 包含 一 个 实现 run() 方 法 的 Runnable 对 象 。 

(3) 调用 上 一 步 创 建 的 Thread 对 和 象 的 start() 方 法 。 
























































示例 如 下 : 

1 public class RunnableThreadExample implements Runnable { 
2 public int count = ©@; 

3 

4 public void run() { 

5 System.out.println(“RunnableThread starting.”); 

6 try { 

7 while (count < 5) { 

8 Thread.sleep(566); 

9 Count++; 

16 

1 } catch (InterruptedException exc) { 

12 System.out.println(“RunnableThread interrupted.”); 
13 } 

14 System.out.println(“RunnableThread terminating.”); 
15 } 

16 } 

17 
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18 public static void main(String[] args) { 


19 RunnableThreadExample instance = new RunnableThreadExample(); 
20 Thread thread = new Thread(instance); 
21 thread. start(); 

22 

23 /* 等 到 上 面 的 线程 数 到 5 (时 间 有 点 长 ) */ 
24 while (instance.count != 5) { 

25 try { 

26 Thread.sleep(250); 

27 } catch (InterruptedException exc) { 
28 exc.printStackTrace(); 

29 } 

36 } 

31 } 








从 上 面 的 代码 可 以 看 出 , 我 们 真正 需要 做 的 是 我 们 的 类 必须 实现 run() 方 法 (第 4 行 ),。 然后 ， 
另 一 个 方法 就 可 以 将 这 个 类 的 实例 传人 new Thread(obj) (第 19~20 行 )， 然 后 调用 那个 线程 的 
start() (第 21 行 )。 

@ 扩展 Thread 类 

创建 线程 还 有 一 种 方式 , 就 是 通过 扩展 Thread 类 实现 。 使 用 这 种 方式 , 基本 上 就 意味 着 要 重 
写 run() 方 法 ， 并 且 在 子 类 的 构造 函数 里 ， 还 需要 显 式 调 用 这 个 线程 的 构造 子 数 。 

下 面 是 使 用 这 种 方式 的 示例 代码 。 

















1 public class ThreadExample extends Thread { 

2 int count = @; 

3 

4 public void run() { 

5 System.out.println(“Thread starting.”); 

6 try { 

7 while (count < 5) { 

8 Thread.sleep(566); 

9 System.out.println(“In Thread, count is ”+ count); 
16 Count++; 

11 

12 } catch (InterruptedException exc) { 

13 System.out.println(“Thread interrupted.”); 
14 } 

25 System.out.println(“Thread terminating.”); 

16 } 

17 } 

18 


19 public class ExampleB { 
26 public static void main(String args[]) { 





21 ThreadExample instance = new ThreadExample(); 
22 instance. start(); 

之 3 

24 while (instance.count != 5) { 

25 try { 

26 Thread.sleep(256); 

27 } catch (InterruptedException exc) { 

28 exc.printStackTrace(); 
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这 段 代码 跟 之 前 的 做 法 非常 相似 。 两 者 的 区 别 在 于 , 既然 我 们 是 扩展 Thread 类 而 非 只 是 实现 
一 个 接口 ， 因 此 可 以 在 这 个 类 的 实例 中 调用 start()。 

@ 扩展 Thread 类 vs. 实现 Runnable 接 口 

在 创建 线程 时 ， 相 比 扩展 Thread 类 ， 实 现 Runnable 接 口 可 能 更 优 ， 理 由 有 二 。 
口 Java 不 支持 多 重 继承 。 因 此 ， 扩 展 Thread 类 也 就 代表 这 个 子 类 不 能 扩展 其 他 类 。 而 实现 
Runnable 接 口 的 类 还 能 扩展 男 一 个 类 。 
口 类 可 能 只 要 求 可 执行 即 可 ， 因 此 ， 继 承 整个 Thread 类 的 开销 过 大 。 

2. 同步 和 锁 

给 定 一 个 进程 内 的 所 有 线程 ,都 共享 同一 存储 空间 ,， 这样 有 好 处 又 有 坏处 。 这 些 线程 就 可 以 
共享 数据 ,非常 有 用 。 不 过 ， 在 两 个 线程 同时 修改 某 一 资源 时 ， 这 也 会 造成 一 些 问题 。Java 提 供 
了 同步 机 制 ， 以 控制 对 共享 资源 的 访问 。 

关键 字 synchronized 和 1ock 构 成 了 代码 同步 执行 的 实现 基础 。 

@ 同步 方法 

最 常见 的 做 法 是 ， 使 用 关键 字 synchronized 对 共享 资源 的 访问 加 以 限制 。 该 关键 字 可 以 用 
在 方法 和 代码 块 上 ， 限 制 多 个 线程 ， 使 之 不 能 同时 执行 同一 个 对 象 的 代码 。 

要 搞 清 楚 最 后 一 点 ， 请 看 以 下 代码 : 


将 
























































1 public class MyClass extends Thread { 

2 private String name; 

3 private MyObject my0bj; 

4 

5 public MyClass(MyObject obj, String n) { 

6 name = nj 

2 my0bj = obj; 

8 

9 

16 public void run() { 

11 my0bj .foo(name); 

12 } 

13 } 

14 

15 public class MyObject { 

16 public synchronized void foo(String name) { 

17 try { 

18 System.out.println(“Thread ”+ name + “.foo(): starting”); 
19 Thread.sleep(3666) ; 

20 System.out.println(“Thread ”+ name + “.foo(): ending”); 
21 } catch (InterruptedException exc) { 

22 System.out.println(“Thread ”+ name + “: interrupted.”); 
23 } 

24 

25 } 
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若 有 两 个 MyClass 实 例 ， 能 否 同时 调用 foo? 这 要 看 情况 ， 若 它们 共用 一 个 Myobject 实 例 ， 
则 答案 是 不 可 以 。 但 是 ， 知 两 个 实例 持 有 不 同 的 引用 , 那么， 答案 就 是 可 以 。 


/* 不 同 的 引用 一 一 两 个 线程 都 能 调用 MyObject.foo() */ 
MyObject obj1 = new MyObject(); 

MyObject obj2 = new MyObject(); 

MyClass thread1 = new MyClass(obj1, “1”); 
MyClass thread2 = new MyClass(o0bj2, “2”); 
thread1. start(); 

thread2.start() 





/* 相同 的 0bj 引 用 。 只 有 一 个 线程 可 以 调用 foo， 另 一 个 线程 必须 等 待 */ 
MyObject obj = new MyObject(); 

MyClass thread1 = new MyClass(obj, “1”); 

MyClass thread2 = new MyClass(obj, “2”); 

thread1.start() 

15 thread2.start() 


静态 方法 会 以 类 锁 〈class lock ) 进行 同步 。 上 面 两 个 线程 无 法 同时 执行 同一 个 类 的 同步 
静态 方法 ， 即 使 其 中 一 个 线程 调用 foo 而 男 一 个 线程 调用 bar 也 不 行 。 


PPAPWUVWOONOUVUAUWOP 
WOPp 


己 
小 


1 public class MyClass extends Thread { 

之 本 

3 public void run() { 

4 if (name.equals(“1”)) MyObject.foo(name); 

5 else if (name.equals(“2”)) MyObject.bar(name); 

6 } 

7 } 

8 

9 public class MyObject { 

16 public static synchronized void foo(String name) { 
11 /* 同 之 前 的 foo 实 现 */ 

12 } 

13 

14 public static synchronized void bar(String name) { 
15 /* 同上 面 的 foo 方 法 */ 

16 } 

主 世 说 


执行 这 段 代 码 ， 打 印 输 出 如 下 : 


Thread 1.foo(): starting 
Thread 1.foo(): ending 
Thread 2.bar(): starting 
Thread 2.bar(): ending 


@ 同步 块 
同样 ， 代 码 块 也 可 以 同步 化 。 其 操作 与 同步 方法 非常 相似 。 





public class MyClass extends Thread { 


1 

2 人 

3 public void run() { 
4 my0bj .foo(name); 
5 


} 
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6 } 

7 public class MyObject { 

8 public void foo(String name) { 
9 synchronized(this) { 


和 同步 方法 一 样 ， 每 个 Myobject 实 例 只 有 一 个 线程 可 以 执行 同步 块 中 的 代码 。 这 就 意味 
着 , 若 thread1 和 thread2 持 有 同一 个 Myobject 实 例 , 那么 ,每 次 只 有 一 个 线程 允许 执行 那个 
代码 块 。 

@ 锁 

若 要 实现 更 细 粒 度 的 控制 ， 我 们 可 以 使 用 锁 (lock )。 锁 (或 监视 器 ) 用 于 对 共享 资源 的 
同步 访问 ， 方 法 是 将 锁 与 共享 资源 关联 在 一 起 。 线 程 必须 先 取得 与 资源 关联 的 锁 ， 才 能 访问 
共享 资源 。 不 管 在 任意 时 间 点 ， 最 多 只 有 一 个 线程 能 拿 到 锁 ， 因 此 ， 只 有 一 个 线程 可 以 访问 

锁 的 常见 用 法 是 ， 从 多 个 地 方 访问 同一 资源 时 ,同一 时 刻 只 有 一 个 线程 才能 访问 。 以 下 面 的 
代码 为 示范 。 
































1 public class LockedATM { 

2 private Lock lock; 

3 private int balance = 1060; 

4 

5 public LockedATM() { 

6 lock = new ReentrantLock(); 

7 } 

8 

9 public int withdraw(int value) { 
16 lock.1lock(); 

11 int temp = balance; 

12 try { 

13 Thread.sleep(166); 

14 temp = temp - value; 

15 Thread.sleep(166); 

16 balance = temp; 

17 } catch (InterruptedException e) { } 
18 lock.unlock(); 

19 return temp; 

26 } 

21 

22 public int deposit(int value) { 
23 lock.1lock(); 

24 int temp = balance; 

25 try { 

26 Thread.sleep(1060); 

27 temp = temp + value; 

28 Thread.sleep(366); 

2 9 balance = temp; 

36 } catch (InterruptedException e) { } 
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31 lock.unlock(); 

32 return temp 

33 } 

34 } 

当然 ， 上述 代 码 做 了 特别 处 理 ， 有 意 降低 了 withdraw ( 提 款 ) 和 deposit ( 存款 ) 的 执行 速 


度 ， 以 便 演 示 可 能 会 出 现 的 问题 。 在 实际 开发 中 , 我们 不 必 写 这 种 代码 , 但 它 反映 的 情况 却 非常 
真实 。 使 用 锁 有 助 于 保护 共享 资源 ， 使 其 免 遭 自 改 。 

3. 死 锁 及 死 锁 的 预防 

死 锁 ( deadlock ) 是 这 样 一 种 情形 : 第 一 个 线程 在 等 待 第 二 个 线程 持 有 的 某 个 对 象 锁 ， 而 第 
二 个 线程 又 在 等 待 第 一 个 线程 持 有 的 对 象 锁 〈 或 是 由 两 个 以 上 线程 形成 的 类 似 情 形 )。 由 于 每 个 
线程 都 在 等 其 他 线程 释放 锁 ， 以致 每 个 线程 都 会 一 直 这 么 等 下 去 。 于 是 , 这些 线程 就 陷入 了 所 谓 
的 死 锁 。 

死 锁 的 出 现 必须 同时 满足 以 下 四 个 条 件 。 

(1) 互 斥 : 某 一 时 刻 只 有 一 个 进程 能 访问 某 一 资源 。( 或 者 ， 更 准确 地 说 ， 对 某 一 资源 的 访问 
有 限制 。 若 资源 数量 有 限 ， 也 可 能 出 现 死 锁 。) 

(2) 持 有 并 等 待 : 已 持 有 某 一 资源 的 进程 不 必 释 放 当 前 拥有 的 资源 ， 就 能 要 求 更 多 的 资源 。 

(3) 没有 抢占 : 一 个 进程 不 能 强制 男 一 个 进程 释放 资源 。 

(4) 循环 等 待 : 两 个 或 两 个 以 上 的 进程 形成 循环 链 ， 每 个 进程 都 在 等 待 循环 链 中 另 一 进程 持 
有 的 资源 。 

若 要 预防 死 锁 ， 只 需 避 免 上 述 任 一 条 件 ， 但 这 很 棘手 ， 因 为 其 中 有 些 条 件 很 难 满足 。 比 如 ， 
想 要 避免 条 件 1 就 很 困难 ， 因 为 许多 资源 同一 时 刻 只 能 被 一 个 进程 使 用 ( 如 打印 机 )。 大 部 分 预防 
死 锁 的 算法 都 把 重心 放 在 避免 条 件 4 即 循环 等 待 上 。 


















































面试 题目 








16.1 线程 和 进程 有 何 区 别 ? ( 第 296 页 ) 

16.2 如 何 测量 上 下 文 切换 时 间 ? (第 296 页 ) 

16.3 在 著名 的 哲学 家 就 餐 问 题 中 , 一 群 哲 学 家 于 坐 在 圆桌 周围 , 每 两 位 哲学 家 之 间 有 一 根 簧 子 。 
每 位 哲学 家 需要 两 根 秘 子 才能 用 和 餐 ， 并且 一 定 会 先 拿 起 左手 边 的 和 纪 子 ， 然 后 才 会 去 拿 右手 
边 的 第 子 。 如 果 所 有 哲学 家 在 同一 时 间 拿 起 左手 边 的 第 子 ， 就 有 可 能 造成 死 锁 。 请 使 用 线 
程 和 锁 ， 编 写 代 码 模拟 哲学 家 就 餐 问题 ， 避 免 出 现 死 锁 。( 第 298 页 ) 

16.4 设计 一 个 类 ， 只 有 在 不 可 能 发 生死 锁 的 情况 下 ， 才 会 提供 锁 。( 第 299 页 ) 

16.5 给 定 以 下 代码 : 


public class Foo { 


























public Foo() { ... } 

public void first() { ... } 

public void second() { ... } 

public void third() { ... } 
} 
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同一 个 Foo 实 例会 被 传人 3 个 不 同 的 线程 。threadA 会 调用 first，threadB 会 调用 second， 
threadC 会 调用 third。 设计 一 种 机 制 , 确保 first 会 在 second 之 前 调用 ,second 会 在 third 
之 前 调用 。( 第 304 页 ) 

16.6 给 定 一 个 类 ， 内 含 同 步 方法 A 和 普通 方法 B。 在 同一 个 程序 实例 中 ， 有 两 个 线程 ， 能 否 同 时 
执行 A? 两 者 能 否 同时 执行 A 和 B? (第 305 页 ) 











8.17 ”中 等 难题 


17.1 编写 一 个 函数 ， 不 用 临时 变量 ， 直 接 交 换 两 个 数 。( 第 306 页 ) 
17.2 设计 一 个 算法 ， 判 断 玩 家 是 否 语 了 井 字 游 戏 。( 第 307 页 ) 
17.3 设计 一 个 算法 ， 算 出 "阶乘 有 多 少 个 尾随 零 。( 第 310 页 ) 
17.4 编写 一 个 方法 ， 找 出 两 个 数字 中 最 大 的 那 一 个 。 不 得 使 用 if-else 或 其 他 比较 运算 符 。 
(第 311 页 ) 
17.5 珠 现 妙 算 游戏 (The Game of Master Mind ) 的 玩法 如 下 。 
计算 机 有 四 个 槽 ， 每 个 槽 放 一 个 球 ， 颜 色 可 能 是 红色 ( R)、 黄 色 (Y )、 绿色 (G ) 或 蓝 
色 (B )。 例 如 ， 计 算 机 可 能 有 RGGB 四 种 〈 覃 1 为 红色 ， 槽 2 、3 为 绿色 ， 覃 4 为 蓝 色 )。 
作为 用 户 ， 你 试图 猜 出 颜色 组 合 。 打 个 比方 ， 你 可 能 会 猜 YRGB。 
要 是 猜 对 某 个 槽 的 颜色 ， 则 算 一 次 “ 猜 中 ”; 要 是 只 猜 对 颜色 但 槽 位 猜 错 了 ， 则 算 一 次 “ 伪 
猜 中 ”。 注 意 ,“ 猜 中 ”不 能 算 入 “ 伪 猜 中 ”。 
举 个 例子 ， 实 际 颜 色 组 合 为 RGBY ， 而 你 猜 的 是 GGRR， 则 算 一 次 猜 中 ， 一 次 伪 猜 中 。 
给 定 一 个 猜测 和 一 种 颜色 组 合 ， 编 写 一 个 方法 ， 返 回 猜 中 和 伪 猜 中 的 次 数 。( 第 313 页 ) 
17.6 给 定 一 个 整数 数组 ， 编 写 一 个 函数 ， 找 出 索引 m 和 7 ， 只 要 将 mm 和 mw 之 间 的 元 素 排 好 序 ， 整 个 
数组 就 是 有 序 的 。 注 意 : n -mm 越 小 越 好 ， 也 就 是 说 ， 找 出 符合 条 件 的 最 短 序列 。 
示例 : 
输入 : 1，2，4，7，16，11，7，12，6，7，16，18，19 
输出 : (3，9) (第 314 页 ) 
17.7 给 定 一 个 整数 ， 打 印 该 整数 的 英文 描述 ( 例如 “One Thousand, Two Hundred Thirty Four”)。 





































































































(第 316 页 ) 
17.8 给 定 一 个 整数 数组 ( 有 正 数 有 负数 )， 找 出 总 和 最 大 的 连续 数列 ， 并 返回 总 和 。 
示例 : 


输入 : 2，-8，3，-2，4，-16 
输出 : 5 ( 即 {3，-2，4} ) (第 318 页 ) 

17.9 设计 一 个 方法 ， 找 出 任意 指定 单词 在 一 本 书 中 的 出 现 频 率 。( 第 319 页 ) 

17.10 XML 非常 兄长 ， 你 找到 一 种 编码 方式 ， 可 将 每 个 标签 对 应 为 预先 定义 好 的 整数 值 ， 该 编 
码 方式 的 语法 如 下 : 
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17.11 


17.12 
17.13 


17.14 


Element > Tag Attributes END Children END 
Attribute --> Tag Value 

END -->0 

Tag -> 映射 至 某 个 预定 义 的 整数 值 

Value > 字符 串 值 END 


例如 ， 下 列 XML 会 被 转换 压缩 成 下 面 的 字符 串 ( 假定 对 应 关系 为 family -> 1、person -> 
2、firstName -> 3、lastName -> 4、state -> 5)。 


<family lastName=“McDowell” state=“CA”> 
<person firstName=“Gayle”>Some Message</persony> 
</family> 


变 为 : 

1 4 McDowell 5 CA 6 2 3 Gayle 6 Some Message 6 6. 

编写 代码 ， 打 印 XML 元素 编码 后 的 版 本 〈 传 人 Element 和 Attribute 对 象 )。( 第 320 页 ) 
给 定 rand5() ， 实 现 一 个 方法 rand7()。 也 即 ， 给 定 一 个 产生 0 到 4 ( 含 ) 随机 数 的 方法 ， 
编写 一 个 产生 0 到 6 ( 含 ) 随机 数 的 方法 。( 第 321 页 ) 

设计 一 个 算法 ， 找 出 数组 中 两 数 之 和 为 指定 值 的 所 有 整数 对 。( 第 323 页 ) 

有 个 简单 的 类 似 结 点 的 数据 结构 BiNode， 包 含 两 个 指向 其 他 结 点 的 指针 : 

1 public class BiNode { 

这 public BiNode node1，node2; 


3 public int data; 
4 】} 


数据 结构 BiNode 可 用 来 表示 二 叉 树 ( 其 中 node1 为 左 子 结 点 ，node2 为 右 子 结 点 ) 或 双向 
链表 ( 其 中 node1 为 前 趋 结 点 , node2 为 后 继 结 点 ), 编 写 一 个 方法 ,将 二 又 查找 树 ( 用 BiNode 
实现 ) 转换 为 双向 链表 。 要 求 所 有 数值 的 排序 不 变 , 转换 操作 不 得 引入 其 他 数据 结构 ( 即 
直接 操作 原先 的 数据 结构 ) ( 第 324 页 ) 

哦 ,不 ! 你 刚刚 写 好 一 篇 长 文 ， 却 倒霉 地 误 用 了 “查找 /替换 ”， 不 慎 删除 了 文档 中 所 有 空 
格 、 标 点 ， 大 写 变 成 小 写 。 比 如 ， 句 子 “I reset the computer. It still didn*t boot!”( 我 重启 
了 电脑 , 但 还 没 启动 好 ! ) 变 成 了 “iresetthecomputeritstilldidntboot”。 你 发 现 ， 只 要 能 正确 
分 离 各 个 单词 ,加 标点 和 调整 大 小 写 都 不 成 问题 。 大 部 分 单词 在 字典 里 都 找 得 到 ， 有 些 字 
符 串 如 专 有 名 词 则 找 不 到 。 

给 定 一 个 字典 (一 组 单词 )， 设 计 一 个 算法 ， 找 出 拆 分 一 连 串 单词 的 最 佳 方式 。 这 里 “最 
佳 ” 的 定义 是 ， 解 析 后 无 法 辨识 的 字符 序列 越 少 越 好 。 

举 个 例子 , 字符 串 “jesslookedjustliketimherbrother” 的 最 佳 解析 结果 为 “JESS looked just like 
TIM her brother”"， 总 共有 7 个 字符 无 法 辨别 ， 全 部 显示 为 大 写 ， 以 示 区 别 。( 第 327 页 ) 






































8.18 ”高 难度 题 


18.1 编写 一 个 函数 ， 将 两 个 数字 相 加 。 不 得 使 用 + 或 其 他 算术 运算 符 。( 第 331 页 ) 
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18.2 编写 一 个 方法 ， 洗 一 副 牌 。 要 求 做 到 完美 洗 牌 ， 换 言 之 ， 这 副 牌 52! 种 排列 组 合 出 现 的 概率 

相同 。 假 设 给 定 一 个 完美 的 随机 数 发 生 器 。( 第 332 页 ) 
18.3 编写 一 个 方法 ， 从 大 小 为 n 的 数组 中 随机 选 出 m 个 整数 。 要 求 每 个 元 素 被 选中 的 概率 相同 。 
(第 333 页 ) 
18.4 编写 一 个 方法 ， 数 出 0 到 n( 含 ) 中 数字 2 出 现 了 几 次 。 
示例 : 
输入 : 25 
输出 : 9 (2，12，26，21，22，23，24 和 25。 注 意 22 有 两 个 2。) (第 334 页 ) 

18.5 有 个 内 含 单词 的 超大 文本 文件 ， 给 定 任意 两 个 单词 ， 找 出 在 这 个 文件 中 这 两 个 单词 的 最 短 
距离 (也 即 相 隔 几 个 单词 )。 有 办 法 在 O(1) 时 间 里 完成 搜索 操作 吗 ? 解法 的 空间 复杂 度 如 
何 ? (第 337 页 ) 

18.6 设计 一 个 算法 ,给 定 10 亿 个 数字 , 找 出 最 小 的 100 万 个 数字 。 假定 计算 机 内 存 足 以 容纳 全 部 
10 亿 个 数字 。( 第 338 页 ) 

18.7 给 定 一 组 单词 ， 编 写 一 个 程序 ， 找 出 其 中 的 最 长 单词 ， 且 该 单词 由 这 组 单词 中 的 其 他 单词 

组 合 而 成 。 
示例 : 
输入 : cat，banana，dog，nana，walk，walker，dogwalker 
输出 : dogwalker (第 339 页 ) 
18.8 给 定 一 个 字符 串 s 和 一 个 包含 较 短 字符 串 的 数组 T, 设计 一 个 方法 , 根据 T 中 的 每 一 个 较 短 字 
符 串 ， 对 s 进 行 搜索 。( 第 341 页 ) 

18.9 随机 生成 一 些 数字 并 传人 某 个 方法 。 编 写 一 个 程序 ， 每 当 收 到 新 数字 时 ， 找 出 并 记录 中 位 
数 。( 第 342 页 ) 

18.10 给 定 两 个 字典 里 的 单词 ， 长 度 相 等 。 编 写 一 个 方法 ， 将 一 个 单词 变换 成 另 一 个 单词 ， 一 
次 只 改动 一 个 字母 。 在 变换 过 程 中 ， 每 一 步 得 到 的 新 单词 都 必须 是 字典 里 存在 的 。 
示例 : 
输入 : DAMP，LIKE 
输出 : DAMP -> LAMP -> LIMP -> LIME -> LIKE (第 343 页 ) 

18.11 给 定 一 个 方 阵 ， 其 中 每 个 单元 ( 像素 ) 非 黑 即 白 。 设 计 一 个 算法 ， 找 出 四 条 边缘 为 黑色 
像素 的 最 大 子 方 阵 。( 第 345 页 ) 

18.12 给 定 一 个 正 整 数 和 负 整 数组 成 的 NxN 和 矩阵 ， 编 写 代码 找 出 元 素 总 和 最 大 的 子 矩 阵 。( 第 
348 页 ) 

18.13 给 定 一 份 几 百 万 个 单词 的 清单 ， 设 计 一 个 算法 ， 创 建 由 字母 组 成 的 最 大 矩形， 其 中 每 一 
行 组 成 一 个 单词 〈 自 左 向 右 )， 每 一 列 也 组 成 一 个 单词 ( 自 上 而 下 )。 不 要 求 这 些 单词 在 
清单 里 连续 出 现 ， 但 要 求 所 有 行 等 长 ， 所 有 列 等 高 。( 第 352 页 ) 
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请 登录 我 们 的 网 站 www.CrackingTheCodingInterview.com， 下 载 完 整 且 可 编译 的 Java/Eclipse 工 
程 , 并 与 其 他 读者 一 起 讨论 书 中 的 面试 题 ， 提 交 间 题 ， 查 看 本 书 勘误 ， 发 布 简历 及 寻求 其 他 建议 。 
数据 结构 
口 数组 与 字符 串 
口 链表 
口 栈 与 队列 
口 树 与 网 
概念 与 算法 
口 位 操作 
口 智力 题 
口 数学 与 概率 
口 面向 对 象 设计 
口 递归 和 动态 规划 
口 扩展 性 与 存储 限制 
口 排序 与 查找 
口 测试 
知识 类 问题 
口 C 和 C++ 
口 Java 
口 数据 库 
口 线程 与 锁 
附加 面试 题 
口 中 等 难题 


口 高 难度 题 














图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 








9.1 数组 与 字符 串 


1.1 实现 一 个 算法 ， 确 定 一 个 字符 串 的 所 有 字符 是 否 全 都 不 同 。 假 使 不 允许 使 用 额外 的 数 
据 结构 ， 又 该 如 何 处 理 ? (第 46 页 ) 

解法 

一 开始 ,不 妨 先 问 问 面试 官 ， 上 面 的 字符 串 是 ASCII 字 符 串 还 是 Unicode 字 符 串 。 这 很 重要 ， 
问 这 个 问题 表明 你 关注 细 广 ， 并 且 对 计算 机 科学 有 深刻 了 解 。 

为 了 简单 起 见 ， 这 里 假定 字符 集 为 ASCII。 若 不 是 的 话 ， 则 需 扩 大 存储 空间 ， 不 过 其 余 逻 辑 
没有 分 别 。 

假定 字符 集 为 ASCI， 对 于 这 个 问题 ， 我 们 可 以 做 一 个 简单 的 优化 ， 若 字符 串 的 长 度 大 于 字 
母 表 中 的 字符 个 数 , 则 直接 返回 false。 毕 竞 , 若 字 母 表 只 有 256 个 字符 , 字符 串 里 就 不 可 能 有 280 
个 各 不 相同 的 字符 。 

第 一 种 解法 是 构建 一 个 布尔 值 的 数组 , 索引 值 i 对 应 的 标记 指示 该 字符 串 是 否 含有 字母 表 第 ii 
个 字符 。 若 这 个 字符 第 二 次 出 现 ， 则 立即 返回 false。 

下 面 是 这 个 算法 的 实现 代码 。 


1 public boolean isUniqueChars2(String str) { 



























































2 if (str.length() > 256) return false; 

3 

4 boolean[] char_set = new boolean[256]; 

5 for (int i = 6;j i < str.length(); i++) { 
6 int val = str.charAt(i); 

7 if (char_set[val]) { // 这 个 字符 已 在 字符 囊 中 出 现 过 
8 return false; 

9 

16 char_set[val] = true; 

11 } 

12 return true; 

13 } 


这 段 代 码 的 时 间 复 杂 度 为 O(n)， 其 中 n 为 字符 串 长 度 。 空 间 复杂 度 为 0(1)。 
使 用 位 向 量 (bit vector )， 可 以 将 空间 占用 减少 为 原先 的 /8。 下 面 的 代码 假定 字符 串 只 含有 
小 写字 母 a 到 z。 这 样 一 来 ， 我 们 只 需 使 用 一 个 int 型 变量 。 


1 public boolean isUniqueChars(String str) { 











2 if (str.length() > 26) return false; 

3 

4 int checker = 0; 

5 for (int i = 6;j i < str.length(); i++) { 
6 int val = str.charAt(i) - ‘a’; 

了 if ((checker & (1 << val)) > 6) { 

8 return false; 

9 

16 checker |= (1 << val); 

pl } 
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12 return true; 
13 } 


另外 ， 还 有 以 下 两 种 解法 。 

( 将 字符 串 中 的 每 一 个 字符 与 其 余 字 符 进行 比较 。 这 种 方法 的 时 间 复 杂 度 为 O(n”)， 空 间 复 
困 度 为 0(1)。 

(2) 若 允 许 修改 输入 字符 串 ， 可 以 在 O(n log(n)) 时 间 里 对 字符 串 排 序 , 然后 线性 检查 其 中 有 无 
相 邻 字符 完全 相同 的 情况 。 不 过 ， 值 得 注意 的 是 ， 很 多 排序 算法 会 占用 额外 的 空间 。 

从 某 些 方面 来 看 ， 这 些 算法 算 不 上 最 优 , 不过， 从 问题 的 限制 条 件 来 看 , 或 许 还 算是 不 错 的 
解法 。 

1.2 用 C 或 C++ 实现 void reverse(char* str) 函 数 ， 即 反 转 一 个 null 结 尾 的 字符 串 。( 第 
46 页 ) 

解法 

这 是 很 经 典 的 面试 题 ， 你 可 能 会 忽略 的 是 : 不 分 配额 外 空间 ， 直 接 就 地 反 转 字符 串 ， 男 外 ， 
还 要 注意 null 字 符 。 

下 面 用 C 语 言 实现 整个 算法 。 


1 void reverse(char *str) { 























2 char* end = str; 

3 char tmp; 

4 if (str) { 

5 while (*end) { /* 找 出 字符 串 末 尾 */ 
6 ++end 

7 

8 --end; /* 回 退 一 个 字符 ， 最 后 一 个 为 hull 字符 */ 
9 

16 /* 从 字符 串 首 尾 开始 交换 两 个 字符 ， *#/ 
11 * 直至 两 个 指针 在 中 间 碰 头 */ 

12 while (str < end) { 

13 tmp = *str; 

14 *str++ = *end; 

了 5 *end-- = tmp; 

16 } 

17 } 

18 } 





上 面 的 代码 只 是 实现 这 个 解法 的 诸多 方法 之 一 。 我 们 甚至 还 可 以 递归 实现 这 段 代码 , 但 并 不 
推荐 这 么 做 。 

1.3 ”给 定 两 个 字符 串 ， 请 编写 程序 ， 确 定 其 中 一 个 字符 串 的 字符 重新 排列 后 ， 能 否 变 成 另 
一 个 字符 串 。( 第 46 页 ) 

解法 

跟 其 他 许多 问题 一 样 ， 首 先 我 们 应 该 向 面试 官 确认 一 些 细节 ,和 弄 清楚 变 位 词 (anagram ) " 比 








Qa 变 位 词 是 由 变换 某 个 词 或 短语 的 字母 顺序 而 构成 的 新 的 词 或 短语 。 一 一 译 者 注 
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较 是 否 区 分 大 小 写 。 比 如 ，God 是 否 为 dog 的 变 位 词 ” 此 外 ,我 们 还 应 该 问 清楚 是 否 要 考虑 空白 
字符 。 

这 里 假定 变 位 词 比 较 区 分 大 小 写 ， 空 日 也 要 考虑 在 内 。 也 就 是 说 ,“god ”不 是 “dog” 的 
变 位 词 。 

比较 两 个 字符 串 时 ， 只 要 两 者 长 度 不 同 ， 就 不 可 能 是 变 位 词 。 

解决 这 个 问题 有 两 个 简单 的 解决 方法 ， 并 且 都 采用 了 上 述 优化 ， 即 先 比 较 字 符 串 长 度 。 

解法 1: 排序 字符 串 

知 两 个 字符 串 互 为 变 位 词 ,， 那么 它们 拥有 同一 组 字符 ， 只 不 过 顺序 不 同 。 因 此 ， 对 字符 串 排 
序 ， 组 成 这 两 个 变 位 词 的 字符 就 会 有 相同 的 顺序 。 我 们 只 需 比 较 排 序 后 的 字符 串 。 
































1 public String sort(String s) { 

2 char[] content = s.toCharArray(); 
3 java.util.Arrays.sort(content); 

4 return new String(content); 
5 

6 

7 

8 


} 


public boolean permutation(String s, String t) { 
if (s.length() != t.length()) { 
9 return false; 
16 } 
1 return sort(s).equals(sort(t)); 








在 茶 种 程度 上 ， 这 个 算法 算 不 上 最 优 , 不 过 换个 角度 看 ， 该 算法 或 许 更 可 取 : 它 清晰 、 简 单 
且 易 懂 。 从 实践 角度 来 看 ， 这 可 能 是 解决 该 问题 的 上 佳之 选 。 

不 过 ， 要 是 效率 当头 ， 我 们 可 以 换 种 做 法 。 

解法 2: 检查 两 个 字符 串 的 各 字符 数 是 否 相 同 

我 们 还 可 以 充分 利用 变 位 词 的 定义 一 一 组 成 两 个 单词 的 字符 数 相 同一 一 来 实现 这 个 算法 。 我 
们 只 需 遍 历 字母 表 ， 计 算 每 个 字符 出 现 的 次 数 。 然 后 ， 比 较 这 两 个 数组 即 可 。 























1 public boolean permutation(String s, String t) { 
2 if (s.length() != t.length()) { 

3 return false; 

4 } 

3 

6 int[] letters = new int[256]; // 假设 条 件 

7 

8 char[] s_array = s.toCharArray(); 

9 for (char c : s_array) { // 计算 字符 串 s 中 每 个 字符 出 现 的 次 数 
16 letters[c]++; 

11 

12 

13 for (int i = 6; i < t.length(); i++) { 

14 int c = (int) t.charAt(i); 

15 if (--letters[c] < 6) { 

16 return false; 
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17 } 

18 } 

19 

20 return true; 
21 } 











注意 第 6 行 的 假设 条 件 。 在 面试 中 ， 最 好 跟 面 试 官 核实 一 下 字符 集 的 大 小 。 这 里 假设 字符 集 
为 ASCII。 


1.4 编写 一 个 方法 ， 将 字符 串 中 的 空格 全 部 替换 为 “%20”。 假 定 该 字符 串 尾部 有 足够 的 
空间 存放 新 增 字符 ， 并 且 知 道 字符 串 的 “真实 ”长 度 。( 注 : 用 Java 实 现 的 话 ， 请 使 用 字符 数组 
实现 ， 以 便 直接 在 数组 上 操作 。) (第 46 页 ) 


解法 

处理 字符 串 操 作 问 题 时 ,常见 做 法 是 从 字符 串 尾部 开始 编辑 ， 从 后 往 前 反 癌 操作。 这 种 做 法 
很 有 用 ， 因 为 字符 串 尾 部 有 额外 的 缓冲 ， 可 以 直接 修改 ， 不 必 担 心 会 覆 写 原 有 数据 。 

我 们 将 采用 上 面 这 种 做 法 。 该 算法 会 进行 两 次 扫描 。 第 一 次 扫描 先 数 出 字符 串 中 有 多 少 空格 ， 
从 而 算出 最 终 的 字符 串 有 多 长 。 第 二 次 扫描 才 真 正 开始 反 向 编辑 字符 串 。 检 测 到 空格 则 将 %26 复 
制 到 下 一 个 位 置 ， 若 不 是 空白 ， 就 复制 原先 的 字符 。 





















































下 面 是 这 个 算法 的 实现 代码 。 

1 public void replaceSpaces(char[] str, int length) { 
2 int spaceCount = 60, newLength, i; 

3 for (i = 6;j i < length; i++) { 

4 if (str[i] == “ *) { 

3 spaceCount++; 

6 } 

6 

8 newLength = length + spaceCount * 2; 
9 str[newLength] = ‘\@’; 

16 for (i = length - 1; i >= 6; i--) { 
11 if (str[i] == “ *) { 

12 str[newLength - 1] = ‘“@’; 

13 str[newLength - 2] = ‘2’ 

14 str[newLength - 3] = ‘%’ 

15 newLength = newLength - 3; 

16 } else { 

17 str[newLength - 1] = str[i]; 
18 newLength = newLength - 1; 

19 } 

20 

21 } 


因为 Java 字 符 串 是 不 可 变 的 ( immutable )， 所 以 我 们 选用 了 字符 数组 来 解决 这 个 问题 。 若 直 
接 使 用 字符 串 ， 返 回 时 就 要 把 字符 串 复 制 一 份 ， 不 过 ， 这 么 做 的 好 处 是 只 需 扫描 一 次 。 
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1.5 “利用 字符 重复 出 现 的 次 数 ， 编 写 一 个 方法 ， 实 现 基本 的 字符 串 压 缩 功能 。 比 如 ， 字 符 
串 aabcccccaaa 会 变 为 a2blc5a3。 若 “压缩 ”后 的 字符 串 没 有 变 短 , 则 返回 原先 的 字符 串 。( 第 
46 页 ) 


解法 
乍 一 看 , 编写 这 个 方法 似乎 相当 简单 ,实则 有 点 复杂 。 我们 会 迭代 访问 字符 串 ， 将 字符 拷贝 
至 新 字符 串 ， 并 数 出 重复 字符 。 这 能 有 多 难 呢 ? 








1 public String compressBad(String str) { 

2 String mystr = “”»; 

3 char last = str.charAt(6); 

4 int count = 1; 

5 for (int i = 1; i < str.length(); i++) { 

6 if (str.charAt(i) == last) { // 找到 重复 字符 
7 
8 


Count++; 

} else { // 插入 字符 的 数目 ， 更 新 last 字 符 
9 mystr += last + “” + Count; 
16 last = str.charAt(i); 
了 count = 1; 
12 } 
13 } 
14 return mystr + last + count; 
15 } 


这 上段 代码 并 未 处 理 压 缩 后 字符 串 比 原始 字符 串 长 的 情况 , 但 除 此 之 外 , 全 都 满足 要 求 。 这 种 
做 法 效率 够 高 吗 ? 不 妨 分 析 一 下 这 段 代码 的 执行 时 间 。 

这 段 代码 的 执行 时 间 为 Oo + 及， 其 中 p 为 原始 字符 串 长 度 ，A 为 
字符 串 为 aabccdeeaa， 则 总 计 有 6 个 字符 序列 。 执 行 速 度 慢 的 原因 是 
度 为 O(n”) (参见 8.1 节 的 StringBuffer 部 分 )。 

我 们 可 以 使 用 stringBuffer 优 化 部 分 性 能 。 


序列 的 数量 。 比 如 ， 若 


字符 
字符 申 拼接 操作 的 时 间 复杂 








1 String compressBetter(String str) { 

2 /* 检查 压缩 后 的 字符 串 是 否 会 变 得 更 长 */ 
3 int size = countCompression(str); 

4 if (size >= str.length()) { 

5 return str; 
6 

7 

8 

9 


} 


StringBuffer mystr = new StringBuffer(); 
char last = str.charAt(@); 

16 int count = 1; 

11 for (int i = 1; i < str.length(); i++) { 


12 if (str.charAt(i) == last) { // 找到 重复 字符 
13 count++; 

14 } else { // 持 入 字符 的 数目 ， 更 新 1ast 字 符 

15 mystr.append(last); // 插入 字符 

16 mystr.append(count); // 插入 数目 

17 last = str.charAt(i); 

18 count = 1; 

19 } 
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20 } 

21 

22 /* 在 上 面 第 15 到 16 行 ， 当 重复 字符 改变 时 ， 
23 * 才 会 桂 入 字符 。 我 们 还 需 在 函数 末尾 更 新 


24 ”* 字符 事 ， 因 为 最 后 一 组 重复 字符 还 未 放 入 
25 * 压缩 字符 囊 中 。 
26 */ 


27 mystr.append(1last); 

28 mystr.append(count); 

29 return mystr.toString(); 

36 } 

31 

32 int countCompression(String str) { 

33 if (str == null || str.isEmpty()) return 6; 
34 char last = str.charAt(@); 

35 int size = 0; 

36 int count = 1; 

37 for (int i = 1; i < str.length(); i++) { 


38 if (str.charAt(i) == last) { 

39 COount++; 

46 } else { 

41 last = str.charAt(i); 

42 size += 1 + String.valueOf(count).length(); 
43 count = 1; 

44 } 

45 } 

46 size += 1 + String.valueOf(count).length(); 
47 return size; 

48 } 


这 个 算法 要 好 得 多 。 注 意 ， 我 们 在 第 2 ~ 5 行 代码 中 加 入 了 长 度 检 查 。 
若 不 想 或 不 能 使 用 stringBuffer， 我 们 还 是 可 以 高 效 地 解决 这 个 问题 。 第 2 行 代码 会 算出 字 
符 串 压缩 后 的 长 度 ， 这 样 就 可 以 构建 出 相应 大 小 的 字符 数组 ， 代 码 实现 如 下 : 
String compressAlternate(String str) { 
/* 检查 压缩 后 的 字符 串 是 否 会 变 得 更 长 */ 
int size = countCompression(str); 


1 

之 

3 

4 if (size >= str.length()) { 
5 return str; 
6 

7 

8 

9 








} 


char[] array = new char[size]; 
int index = 0@; 
16 char last = str.charAt(@); 
TL int count = 1; 
12 for (int i = 1; i < str.length(); i++) { 


13 if (str.charAt(i) == last) { // 找到 重复 字符 
14 count++; 

15 } else { 

16 /* 更 新 重复 字符 的 数目 */ 

17 index = setChar(array, last, index, count); 
18 last = str.charAt(i); 

19 count = 1; 
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26 } 

21 } 

22 

23 /* 以 最 后 一 组 重复 字符 更 新 字符 串 */ 

24 index = setChar(array, last, index, count); 
25 return String.valueOf(array); 

26 } 

27 

28 int setChar(char[] array, char c, int index, int count) { 
29 array[index] = cj 

36 index++; 

31 


32 /* 将 数目 转换 成 字符 事 ， 然 后 转 成 字符 数组 */ 
33 char[] cnt = String.valueOf(count).toCharArray(); 


35 /* 从 最 大 的 数字 到 最 小 的 ， 复 制 字符 */ 
36 for (char x : cnt) { 


37 array[index] = Xx; 
38 index++; 

39 } 

46 return index; 

41 } 

42 


43 int countCompression(String str) { 
44 /* 与 之 前 实现 相同 */ 
45 } 


跟 第 二 种 解法 一 样 ， 执 行 上 述 代 码 的 时 间 复 杂 度 为 O(N)， 空 间 复杂 度 为 O(N)。 

1.6 ”给 定 一 幅 由 NxN 和 矩阵 表示 的 图 像 ， 其 中 每 个 像素 的 大 小 为 4 字 节 ,编写 一 个 方法 ， 将 
图 像 旋转 90 度 。 不 占用 额外 内 存 空间 能 否 做 到 ? (第 47 页 ) 

解法 

要 将 矩阵 旋转 90 度 ,最 简单 的 做 法 就 是 一 层 一 层 进行 旋转 。 对 每 一 层 执行 环 状 旋转 ( circular 
rotation )， 将 上 边 移 到 右边 、 右 边 移 到 下 边 、 下 边 移 到 左边 、 左 边 移 到 上 边 。 
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那么 ， 该 如 何 交换 这 四 条 边 ? 一 种 做 法 是 把 上 面 复制 到 一 个 数组 中 ， 然 后 将 左边 移 到 上 边 
下 边 移 到 左边 ， 等 等 。 这 需要 占用 O(N) 内 存 空间 ， 实 际 上 没有 必要 。 
更 好 的 做 法 是 按 索 引 一 个 一 个 进行 交换 ， 具 体 做 法 如 下 : 








1 fori=6ton 

2 temp = top[i]; 

3 top[i] = left[i] 

4 left[i] = bottom[i] 
5 bottom[i] = right[i] 
6 right[i] = temp 





从 最 外 面 一 层 开始 逐渐 向 里 ， 在 每 一 层 上 执行 上 述 交换 。( 另外 ， 也 可 以 从 内 层 开始 ， 逐 层 
向 外 。) 
下 面 是 该 算法 的 实现 代码 。 




















1 public void rotate(int[][] matrix, int n) { 

2 for (int layer = 6; layer < n / 2; ++layer) { 
3 int first = layer; 

4 int last =n - 1 - layer; 

5 for(int i = first; i < last; ++i) { 

6 int offset = i - first; 

7 // 存储 上 边 

8 int top = matrix[first][i]; 





9 

16 // 左 到 上 

11 matrix[first][i] = matrix[last-offset][first]; 
12 

13 // 下 到 左 

14 matrix[last-offset][first] = matrix[last][last - offset]; 
15 

16 // 右 到 下 

17 matrix[last][last - offset] = matrix[i][last]; 
18 

19 // 上 到 右 

20 matrix[i][last] = top; 

21 } 

22 } 

23 让 





这 个 算法 的 时 间 复 杂 度 为 OOV) , 这 已 是 最 优 的 做 法 , 因为 任何 算法 都 需要 访问 所 有 和 个 元 素 。 

1.7 ”编写 一 个 算法 ， 若 MxN 和 矩阵 中 某 个 元 素 为 0， 则 将 其 所 在 的 行 与 列 清 零 。( 第 47 页 ) 

解法 

bl 这 个 问题 似乎 很 简单 : 直接 遍历 整个 矩阵 ， 只 要 发 现 值 为 零 的 元 素 ， 就 将 其 所 在 的 
行 与 列 清 零 。 不 过 这 种 方法 有 个 陷阱 : 在 读 取 被 清 零 的 行 或 列 时 , 读 到 的 尽 是 零 ， 于 是 所 在 行 与 
列 都 ， 。 很 快 ， 整 个 矩阵 的 所 有 元 素 都 会 变 为 堆 


各 开 这 个 陷阱 的 方法 之 _， 是 新 建 一 个 矩阵 标记 零 元 素 位 置 。 然后， 在 第 二 次 遍历 矩阵 时 将 
零 元 素 所 在 行 与 列 清 零 。 这 种 做 法 的 空间 复杂 度 为 OUMNV)。 9 
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真 的 需要 占用 O(MN) 空 间 吗 ? 不 是 的 。 既 然 打算 将 整 行 和 整 列 清 为 零 ,因此 并 不 需要 准确 记 


录 它 是 cel1[2][4]《〈 行 2、 列 4 )， 只 需 知 道行 2 有 个 元 素 为 零 ， 列 4 有 个 元 素 为 零 。 不 管 怎 样 ， 整 











行 和 整 列 都 要 清 为 零 ， 又 何必 要 记录 零 元 素 的 确切 位 置 ? 





下 面 是 这 个 算法 的 实现 代码 。 这 里 用 两 个 数组 分 别 记 录 包 含 零 元 素 的 所 有 行 和 列 。 然 后 , 在 








第 二 次 遍历 矩阵 时 ， 若 所 在 行 或 列 标记 为 零 ， 则 将 元 素 清 为 零 。 


1 public void setzeros(int[][] matrix) { 

2 boolean[] row = new boolean[matrix.1length]; 

3 boolean[] column = new boolean[matrix[8].length]; 
4 

5 // 记录 值 为 6 的 元 素 所 在 行 索引 和 列 索 引 

6 for (int i = 6; i < matrix.length; i++) { 

7 for (int j = 60; j < matrix[6].length;j++) { 
8 if (matrix[i][j] == 6) { 

9 row[i] = true; 

16 column[j] = true; 

11 

12 } 

13 } 

14 


15 // 苦行 i 或 列 j 有 个 元 素 为 8， 则 将 arr[i][j] 置 为 6 

16 for (int i = 6; i < matrix.length; i++) { 

17 for (int j = 6; j < matrix[6].length; j++) { 
18 if (row[i] || column[j]) { 

19 matrix[i][j] = 6; 

26 } 

21 } 


为 了 提高 空间 利用 率 ， 我 们 可 以 选用 位 向 量 替代 布尔 数组 。 


1.8 假定 有 一 个 方法 ijssubstring， 可 检查 一 个 单词 是 否 为 其 他 字符 串 的 子 串 。 给 定 两 个 字 


符 串 s1 和 s2， 请 编写 代码 检查 s2 是 否 为 s1 旋转 而 成 ， 要 求 只 能 调用 一 次 issubstring。( 比 如 ， 
waterbottle 是 erbottlewat 旋 转 后 的 字符 串 。 ) (第 47 页 ) 


解法 
假定 s2 由 si 旋转 而 成 ,那么 ， 我们 可 以 找 出 旋转 点 在 哪 。 例 如 ， 阁 以 wat 对 waterbottle 旋 





转 ， 就 会 得 到 erbottlewat。 在 旋转 字符 串 时 ， 我 们 会 把 s1 切 分 为 两 部 分 : x 和 y， 并 将 它们 重新 
组 合成 s2。 


sl1 = xy = waterbottle 
x = wat 

y = erbottle 

Ss2 = yx = erbottlewat 


因此 ， 我 们 需要 确认 有 没有 办 法 将 s1 切 分 为 x- 和 y， 以 满足 xy = sl 和 yx = s2。 不 论 x 和 yy 之 








间 的 分 割 点 在 何 处 ， 我 们 会 发 现 yx 肯 定 是 xyxy 的 子 串 。 也 即 ，s2 总 是 s1s1 的 子 串 。 
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上 述 分 析 正 是 这 个 问题 的 解法 : 直接 调用 issubstring(s1s1，s2) 即 可 。 
下 面 是 上 述 算法 的 实现 代码 。 























public boolean isRotation(String s1, String s2) { 
int len = si1.length(); 
/* 检查 S1 和 S2 是 否 等 长 且 不 为 空 */ 
if (len == s2.length() && len > 6) { 
/* 拼接 sS1 和 Ss1， 放 入 新 字符 串 中 */ 
String S1s1 = sl + S1; 
return isSubstring(s1s1, s2); 


1 
2 
3 
4 
5 
6 
到 
8 
9 return false; 
1 


8 } 
9.2 ”链表 


2.1 编写 代码 ， 移 除 未 排序 链表 中 的 重复 结 点 。 


进 阶 
如 果 不 得 使 用 临时 缓冲 区 ， 该 怎么 解决 9 ( 第 48 页 ) 
解法 


要 想 移 除 链表 中 的 重复 结 点 , 我 们 需要 设法 记录 有 哪些 是 重复 的 。 这 里 只 要 用 到 一 个 简单 的 
散 列 表 。 
在 下 面 的 解法 中 , 我 们 会 直接 迭代 访问 整个 链表 , 将 每 个 结 点 加 入 散 列 表 。 知 发 现 有 重复 元 





素 ， 则 将 该 结 点 从 链表 中 移 除 ， 然 后 继续 迭代 。 这 个 题目 使 用 了 链表 ， 因 此 只 需 扫 描 一 次 就 能 
搞定 。 

1 public static void deleteDups(LinkedListNode n) { 

2 Hashtable table = new Hashtable(); 

3 LinkedListNode previous = null; 

4 while (n != null) { 

5 if (table.containsKey(n.data)) { 

6 previous.next = n.next; 

7 } else { 

8 table.put(n.data, true); 

9 previous = nj 

16 } 

11 n = n.next; 

12 } 

13. 上 


上 述 代码 的 时 间 复 杂 度 为 O(N)， 其 中 入 为 链表 结 点 数目 。 


进 阶 : 不 得 使 用 缓冲 区 
如 不 借助 额外 的 缓冲 区 ， 可 以 用 两 个 指针 来 迭代 : current 人 迭代 访问 整个 链表 ，runner 用 于 


检查 后 续 的 结 点 是 否 重复 。 9 
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1 public static void deleteDups(LinkedListNode head) { 
2 if (head == null) return; 
3 
4 LinkedListNode current = head; 
5 while (current != null) { 
6 /* 移 除 后 续 值 相同 的 所 有 结 点 */ 
7 LinkedListNode runner = current; 
8 while (runner.next != null) { 
9 if (runner.next.data == current.data) { 
16 runner.next = runner.next.next; 
11 } else { 
12 runner = runner.next; 
13 } 
14 } 
15 current = current.next; 
16 } 
17 } 


这 段 代码 的 空间 复杂 度 为 0(1)， 但 时 间 复 杂 度 为 O(N”)。 

2.2 ”实现 一 个 算法 ， 找 出 单 向 链表 中 倒数 第 k 个 结 点 。( 第 48 页 ) 

解法 

下 面 会 以 递归 和 非 递 归 的 方式 解决 这 个 问题 。 一 般 来 说 ， 递 归 解 法 更 简洁 ， 但 效率 比较 差 。 
例如 ， 就 这 个 问题 来 说 ， 递 归 解 法 的 代码 量 大 概 只 有 迭代 解法 的 一 半 ,， 但 要 占用 OU 空间， 其 中 
1 为 链表 结 点 个 数 。 

注意 ， 在 下 面 的 解法 中 ,Kk 定义 如 下 : 传人 k = 1 将 返回 最 后 一 个 结 点 ，k = 2 返回 倒数 第 2 
个 结 点 ， 依 此 类 推 。 当 然 ， 也 可 以 将 k 定 义 为 K = 68 返回 最 后 一 个 结 点 。 

解法 1: 链表 长 度 已 知 

若 链 表 长 度 已 知 ， 那 么 ， 倒 数 第 kK 个 结 点 就 是 第 (length - k) 个 结 点 。 直 接 迭 代 访 问 链 表 就 
能 找到 这 个 结 点 。 不 过 ， 这 个 解法 太 过 简单 了 ， 不 大 可 能 是 面试 官 想 要 的 答案 。 








解法 2: 递归 

这 个 算法 会 递归 访问 整个 链表 ， 当 抵达 链表 末端 时 ， 该 方法 会 回 传 一 个 置 为 0 的 计数 器 。 
之 后 的 每 次 调用 都 会 将 这 个 计数 器 加 1。 当 计数 器 等 于 时， 表示 我 们 访问 的 是 链表 倒数 第 个 
元 素 。 

实现 代码 简洁 明了 ， 前 提 是 我 们 要 有 办 法 通过 栈 “ 回 传 ” 一 个 整数 值 。 可 惜 ， 我 们 无 法 用 一 
般 的 返回 语句 回 传 一 个 结 点 和 一 个 计数 器 ， 那 该 怎么 办 ? 

@ 方法 A: 不 返回 该 元 素 

一 种 方法 是 对 这 个 问题 略 作 调整 ， 只 打印 倒数 第 个 结 点 的 值 。 然 后 ， 直 接 通 过 返回 值 传 回 
计数 需 值 。 

1 public static int nthToLast(LinkedListNode head, int k) { 


2 if (head == null) { 
3 return 0@; 
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4 
5 int i = nthToLast(head.next, k) + 1; 
6 if (i == K) { 

7 System.out.println(head.data); 

8 

9 


} 
return i; 
16 } 
当然 ， 只 有 得 到 面试 官 的 首肯 ， 这 个 解法 才 算 有 效 。 


@ 方法 B: 使 用 C++ 
第 二 种 解法 是 使 用 C++， 并 通过 引用 传 值 。 这 样 一 来 ， 我 们 就 可 以 返回 结 点 值 ， 而 且 也 能 ; 
过 传递 指针 更 新 计数 器 。 




















1 node* nthToLast(node* head, int k, int& i) { 
之 if (head == NULL) { 

3 return NULL; 

4 

5 node * nd = nthToLast(head->next, k, i); 
6 i=i+1; 

7 if (i == kK) { 

8 return head; 

9 } 

16 return nd ; 

11 } 

e@ 方法 C: 创建 包 训 类 

前 面 提 到 ,这 里 的 难点 在 于 我 们 无 法 同时 返回 计数 器 和 索引 值 。 如 果 用 一 个 简单 的 类 (或 一 





个 单元 素数 组 ) 包 右 计数 器 值 ， 就 可 以 模仿 按 引 用 传递 。 


1 public class Intwrapper { 

2 public int value = 0@; 

3 

4 

5 LinkedListNode nthToLastR2(LinkedListNode head, int k, 
6 Intwrapper i) { 

7 if (head == null) { 

8 return null; 

9 } 

16 LinkedListNode node = nthToLastR2(head.next, k, i); 
11 i.value = i.value + 1; 

12 if (i.value == k) { // 找到 倒数 第 k 个 元 素 

13 return head; 

14 } 

25 return node; 

16 } 

17 


因为 有 递归 调用 ， 这 些 递归 解法 都 需要 占用 O(n) 空 间 。 

还 有 不 少 其 他 解法 这 里 并 未 提 及 。 我 们 可 以 将 计数 器 存放 在 静态 变量 中 , 或 者 ,可 以 创建 一 
个 类 , 存放 结 点 和 计数 器， 并 返回 这 个 类 的 实例 。 不 论 选用 哪 种 解法 ,我 们 都 要 设法 更 新 结 点 和 
计数 器 ， 并 在 每 层 递 归 调 用 的 栈 都 能 访问 到 。 
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解法 3: 和 迭代 法 

一 种 效率 更 高 但 不 太 直观 的 解法 是 以 迭代 方式 实现 。 我 们 可 以 使 用 两 个 指针 p1 和 p2, 并 将 它 
们 指向 链表 中 相距 [个 结 点 的 两 个 结 点 ， 具 体 做 法 是 先 将 p1 和 p2 指 向 链表 首 结 点 ， 然 后 将 p2 向 前 
移动 kf 个 结 点 。 之 后 ， 我 们 以 相同 的 速度 移动 这 两 个 指针 ，p2 会 在 移动 LENGTH - k 步 后 抵达 链表 
尾 结 点 。 此 时 ，p1 会 指向 链表 第 LENGTH - k 个 结 点 , 或 者 说 倒数 第 个 结 点 。 

下 面 的 代码 实现 了 该 算法 。 











1 LinkedListNode nthToLast(LinkedListNode head, int k) { 
2 if (k <= 6) return null; 

3 

4 LinkedListNode pl1 = head; 

5 LinkedListNode p2 = head; 

6 

7 // p2 向 前 移动 k 个 结 点 

8 for (int i = 6ji<k- 1; i++) { 

9 if (p2 == null) return null; // 错误 检查 
16 p2 = p2.next; 

11 

2 if (p2 == null) return null; 

13 

14 /* 现在 以 同样 的 速度 移动 pl 和 p2， 当 p2 抵 达 链 表 末 尾 时 ， 
15 * p1 刚 好 指向 倒数 第 Kk 个 结 点 */ 

16 while (p2.next != null) { 

到 pl = p1.next; 

18 p2 = p2.next; 

19 

20 return pl1; 

21 } 


这 个 算法 的 时 间 复 杂 度 为 0(n)， 空 间 复 杂 度 为 0(1)。 

2.3 ”实现 一 个 算法 ， 删 除 单 向 链表 中 间 的 某 个 结 点 ， 假 定 你 只 能 访问 该 结 点 。( 第 48 页 ) 

解法 

在 这 个 问题 中 ,你 访问 不 到 链表 的 首 结 点 ， 只 能 访问 那个 待 删除 结 点 。 解 法 很 简单 ， 直 接 将 
后 继 结 点 的 数据 复制 到 当前 结 点 ， 然 后 删除 这 个 后 继 结 点 。 

下 面 是 该 算法 的 实现 代码 。 








1 public static boolean deleteNode(LinkedListNode n) { 
2 if (n == null || n.next == null) { 

3 return false; // 失败 

4 } 

5 LinkedListNode next = n.next; 

6 n.data = next.data; 

7 n.next = next.next; 

8 return true; 

9 } 


注意 , 若 待 删除 结 点 为 链表 的 尾 结 点 ， 则 这 个 问题 无 解 。 没 关系 ,面试 官 就 是 想 要 你 指出 这 
一 点 ， 并 讨论 该 怎么 处 理 这 种 情况 。 例 如 ， 你 可 以 考虑 将 该 结 点 标记 为 假 的 。 
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2.4 编写 代码 ， 以 给 定 值 x 为 基准 将 链表 分 割 成 两 部 分 ， 所 有 小 于 x 的 结 点 排 在 大 于 或 等 于 x 
的 结 点 之 前 。( 第 49 页 ) 


解法 

要 是 链表 换 作 数组 ， 搬 移 元 素 时 就 要 特别 小 心 ， 因 为 搬移 数组 元 素 的 开销 很 大 。 

不 过 ,移动 链表 的 元 素 则 要 容易 许多 。 我 们 不 必 移 动 和 交换 元 素 ， 可 以 直接 创建 两 个 链表 : 
一 个 链表 存放 小 于 x 的 元 素 ; 另 一 个 链表 存放 大 于 或 等 于 x 的 元 素 。 

我 们 会 迭代 访问 整个 链表 ， 将 元 素 搬 人 和 人 before 或 after 链 表 。 一 旦 抵达 链表 未 端 ， 则 表明 拆 
分 完成 ， 最 后 合并 两 个 链表 。 

下 面 是 该 方法 的 实现 代码 。 

1 /* 传 入 链表 的 首 结 点 ， 以 及 作为 链表 分 割 

2 * 基准 的 值 */ 
3 public LinkedListNode partition(LinkedListNode node, int x) { 
4 LinkedListNode beforeStart = null; 
3 
6 
了 
8 














LinkedListNode beforeEnd = null; 
LinkedListNode afterStart = null; 
LinkedListNode afterEnd = null; 


9 /* 分 割 链表 */ 
16 while (node != null) { 


LL LinkedListNode next = node.next; 
12 node.next = null; 

13 if (node.data < x) { 

14 /* 将 结 点 插入 before 链 表 */ 
15 if (beforeStart == null) { 
16 beforestart = node; 

17 beforeEnd = beforeStart; 
18 } else { 

19 beforeEnd.next = node; 
20 beforeEnd = node; 

21 } 

22 } else { 

23 /* 将 结 点 插入 after 链 表 */ 
24 if (afterStart == null) { 
25 afterStart = node; 

26 afterEnd = afterStart; 
27 } else { 

28 afterEnd.next = node; 
29 afterEnd = node; 

36 } 

31 } 

32 node = next; 

33 } 

34 

35 if (beforeSstart == null) { 

36 return afterStart; 

37 } 

38 


39 /* 合并 before 和 after 链 表 */ 
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46 beforeEnd.next = afterStart; 
41 return beforestart ; 
42 } 


为 了 追踪 两 个 链表 却 要 维护 四 个 变量 , 你 可 能 觉得 有 点 人 碍 眼 , 不 少 人 都 有 同感 。 我 们 可 以 移 
除 其 中 部 分 变量 , 不 过 代码 执行 效率 会 略 打折 扣 。 效率 降低 的 原因 在 于 遍历 整个 链表 的 时 间 略 微 
延长 。 不 过 ， 大 0 表示 的 时 间 复 杂 度 仍 保持 不 变 ， 同 时 代码 也 变 得 更 简短 、 扼 要 。 

第 二 种 解法 略 有 不 同 。 结 点 不 再 追加 至 before 和 after 链 表 的 末端 ， 而 是 插入 这 两 个 链表 
的 前 端 。 











1 public LinkedListNode partition(LinkedListNode node, int x) { 
2 LinkedListNode beforeStart = null; 

3 LinkedListNode afterStart = null; 

4 

5 /* 分 割 链表 */ 

6 while (node != null) { 

4 LinkedListNode next = node.next; 

8 if (node.data < x) { 

9 /* 将 结 点 插入 before 链 表 的 前 痛 */ 
16 node.next = beforeStart; 

六 于 beforestart = node; 


12 } else { 


13 /* 将 结 点 插入 after 链 表 的 前 端 */ 
14 node.next = afterStart; 

5 afterSstart = node; 

16 } 

17 node = next; 

18 } 

19 

26 /* 合并 before 链 表 和 after 链 表 */ 


21 if (beforeStart == null) { 

22 return afterStart; 

23 } 

24 

25 /* 定位 至 before 链 表 末 尾 ， 合 并 两 个 链表 */ 
26 LinkedListNode head = beforeStart; 

27 while (beforeStart.next != null) { 


28 beforestart = beforeStart.next; 
29 } 

36 beforestart .next = afterStart; 

3 

32 return head; 

33 } 


注意 ， 解 决 这 个 问题 时 ， 必 须 非 常 小 心地 处 理 nul1 值 。 再 看 看 上 面 第 7 行 代码 ， 为 什么 要 有 
这 行 代码 ? 这 是 因为 在 循环 访问 链表 时 , 也 会 修改 这 个 链表 。 我 们 必须 用 临时 变量 记 下 后 继 结 点 ， 
这 样 才能 知道 下 一 次 迭代 要 用 到 该 后 继 结 点 。 
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2.5 ”给 定 两 个 用 链表 表示 的 整数 ， 每 个 结 点 包含 一 个 数位 。 这 些 数 位 是 反 向 存放 的 ， 也 就 
是 个 位 排 在 链表 首部 。 编 写 图 数 对 这 两 个 整数 求 和 ， 并 用 链表 形式 返回 结果 。 

进 阶 

假设 这 些 数 位 是 正 向 存放 的 ， 请 再 做 一 遍 。( 第 49 页 ) 


解法 
着 手 解决 这 个 问题 之 前 ， 有 必要 回顾 一 下 加 法 是 怎么 回 事 ， 比 如 : 





本 
+295 
首先 ，7 加 5 得 到 12。 其 中 ，2 为 结果 12 的 个 位 ，1 则 为 十 位 相 加 时 的 进位 。 然 后 ， 将 1 、1 和 9 
相 加 ， 得 到 11。 十 位 数字 为 1， 另 一 个 1 则 成 为 下 一 步 运 算 的 进位 。 最 后 ， 将 1、6 和 2 相 加 得 到 9。 
因此 ， 这 两 个 整数 求 和 的 结果 为 912。 
我 们 可 以 递归 地 模拟 这 个 过 程 ,将 两 个 结 点 的 值 逐 一 相 加 ， 如 有 进位 则 转 人 下 一 个 结 点 。 下 
面 以 两 个 链表 为 例 进行 说 明 : 
7->1->6 
+5->9->2 
步骤 如 下 。 
(1) 首先 ， 将 7 和 5 相 加 ， 结 果 为 12， 则 ?成 为 结果 链表 的 第 一 个 结 点 ， 并 将 1 进位 给 下 一 次 求 
和 运算 。 
链表 : 2 -> ? 
(2) 然 后 ,将 1、9 和 上 面 的 进位 相 加 ， 结 果 为 11， 于 是 1 成 为 结果 链表 的 第 二 个 元 素 ， 另 一 个 
1 则 进位 给 下 一 个 求 和 运算 。 
链表 : 2 -> 1 -> ? 
(3) 最 后 ， 将 6、2 和 上 面 的 进位 相 加 ， 得 到 9， 同 时 也 成 为 结果 链表 的 最 后 一 个 元 素 。 
链表 : 2 -> 1 -> 9 
下 面 是 该 算法 的 实现 代码 。 


1 LinkedListNode addLists(LinkedListNode 11，LinkedListNode 12， 
int carry) { 
































3 /* 两 个 链表 全 部 为 空 且 进位 为 9， 则 函数 返回 */ 

4 if (11 == null && 12 == null && carry == 8) { 
5 return null; 

6 } 

yd 

8 LinkedListNode result = new LinkedListNode(); 
9 

16 /* 将 value 以 及 11 和 12 的 data 相 加 */ 

11 int value = carry; 

12 if (11 != null) { 

13 value += 11.data; 
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14 } 

15 if (12 != null) { 
16 value += 12.data; 
17 } 

18 


19 result.data = value % 16; /* 求 和 结果 的 个 位 */ 


21 /* 递归 */ 


22 LinkedListNode more = addLists(]1 == null ? null : 11.next, 
23 12 == null ? null : 12.next, 
24 value >= 106 ?1 : 0); 

25 result.setNext(more); 

26 return result; 

27 } 


在 实现 这 段 代 码 时 , 务必 注意 处 理 一 个 链表 比 男 一 个 链表 结 点 少 的 情况 。 我 们 可 不 想 碰 到 空 


首 针 异 常 。 


进 阶 





从 概念 上 来 说 ， 第 二 部 分 并 无 不 同 ( 递归， 进位 处 理 )， 但 在 实现 时 稍微 复杂 一 些 。 


(1) 一 个 链表 的 结 点 可 能 比 男 一 个 链表 的 少 ， 


我 们 无 法 直接 处 理 





这 种 情况 。 例 如 





， 假 设 要 对 


(1 -> 2 -> 3 -> 4) 与 (5 -> 6 -> 7) 求 和 。 务 必 注 意 ，5 应 该 与 2 而 不 是 1 配对 。 对 此 ， 我 们 可 














以 一 开始 先 比较 两 个 链表 的 长 度 并 用 零 填 充 较 短 的 链表 。 

(2) 在 前 一 个 问题 中 ， 相 加 的 结果 不 断 追 加 到 链表 尾部 ( 也 即 向 前 传递 )。 这 就 意味 着 递归 调 
用 会 传人 进位 ， 而 且 会 返回 结果 ( 随后 追加 至 链表 尾部 )。 不 过 ， 这 里 的 结果 要 加 到 首部 ( 也 即 
向 后 传递 ) 跟前 一 个 问题 一 样 ， 递 归 调用 必须 返回 结果 和 进位 。 实 现 也 不 是 太 难 ， 
会 更 难 一 些 ， 可 以 通过 创建 一 个 Partialsum 包 囊 类 来 解决 这 一 点 。 











下 面 是 该 算法 的 实现 代码 。 


public class PartialSum { 
public LinkedListNode sum = null; 
public int carry = 9; 


于 

2 

3 

4 } 
5 

6 

7 int len1 = length(11); 

8 int len2 = length(12); 

16 /* 用 堆 填 充 较 短 的 链表 ， 参 看 注解 (1) */ 
11 if (lenl < len2) { 


2 11 = padList(11, len2 - len1); 
13 } else { 

14 12 = padList(12，]len1 - len2); 
15 } 

16 


17 /* 对 两 个 链表 求 和 */ 





18 PartialSum sum = addListsHelper(11, 12); 


26 /* 如 有 进位 ， 则 插入 链表 首部 ， 否 则 ， 直 接 返 回 
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LinkedListNode addLists(LinkedListNode 11, LinkedListNode 12) { 





日 处 理 起 来 
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28 } 


* 整个 链表 */ 
if (sum.carry == 86) { 
return sum.sum; 
} else { 
LinkedListNode result = insertBefore(sum.sum, sum.carry); 
return result; 


} 


36 PartialSum addListsHelper(LinkedListNode 11, LinkedListNode 12) { 


33 


48 } 


if (11 == null && 12 == null) { 
PartialSum sum = new PartialSum(); 
return sum; 


} 
/* 对 较 小 数字 递归 求 和 */ 
PartialSum sum = addListsHelper(l1.next, 12.next); 


/* 将 进位 和 当前 数据 相 加 */ 
int val = sum.carry + 11.data + 12.data; 


/* 插入 当前 数字 的 求 和 结果 */ 
LinkedListNode full_ result = insertBefore(sum.sum, val % 10); 


/* 返回 求 和 结果 和 进位 值 */ 
sum.sum = full_result; 
sum.carry = val / 108; 
return sum; 


58 /* 用 零 填充 链表 */ 
51 LinkedListNode padList(LinkedListNode 1, int padding) { 


66 } 


LinkedListNode head = 1; 
for (int i = 8; i < padding; i++) { 
LinkedListNode n = new LinkedListNode(6@, null, null); 
head.prev = n; 
n.next = head ; 
head = nj; 
} 


return head; 


62 /* 辅助 函数 ， 将 结 点 插入 链表 首部 */ 
63 LinkedListNode insertBefore(LinkedListNode list, int data) { 


67 


76 } 


LinkedListNode node = new LinkedListNode(data, null, null); 
if (list != null) { 

list.prev = node; 

node.next = list; 


} 


return node; 


注意 ， 上 面 的 代码 已 将 insertBefore() 、padList() 和 1length() (未 列 出 ) 单列 为 独立 方 
法 。 这 样 一 来 ， 代 码 更 清晰 更 易 读 ， 在 面试 时 这 么 做 非常 可 取 ! 
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2.6 ”给 定 一 个 有 环 链 表 ， 实 现 一 个 算法 返回 环 路 的 开头 结 点 。( 第 49 页 ) 


解法 
这 个 问题 是 由 经 典 面试 题 一 一 检测 链表 是 否 存 在 环 路 一 一 演变 而 来 。 下 面 我 们 将 运用 模式 匹 
配 法 来 解决 这 个 问题 。 


第 1 部 分 : 检测 链表 是 否 存在 环 路 

检测 链表 是 否 存在 环 路 ， 有 一 种 简单 的 做 法 叫 FastRunner/SlowRunner 法 。FastRunner 一 次 
移动 两 步 ， 而 SlowRunner 一 次 移动 一 步 。 这 就 好 比 两 辆 赛车 绕 着 同一 条 赛 道 以 不 同 的 速度 前 进 ， 
最 终 必然 会 碰 到 一 起 。 

聪明 的 读者 可 能 会 问 : FastRunner 会 不 会 刚好 “越过 ”SlowRunner ， 而 不 发 生 碰 撞 呢 ? 绝 
无 可 能 。 假 设 FastRunner 真 的 越过 了 SlowRunner， 且 SlowRunner 处 于 位 置 ，FastRunner 处 于 
位 置 i + 1。 那 么 ， 在 前 一 步 ，slowRunner 就 处 于 位 置 i - 1，FastRunner 处 于 位 置 (( + 1) - 
2) 或 i - 1。 也 就 是 说 ， 两 者 碰 在 一 起 了 。 


第 2 部分: 什么 时 候 碰 在 一 起 ? 

假定 这 个 链表 有 一 部 分 不 存在 环 路 ， 长 度 为 k。 

车 运用 第 1 部 分 的 算法 ，FastRunner 和 SlowRunner 什 么 时 候 会 碰 在 一 起 呢 ? 

我 们 知道 ，SlowRunner 每 走 p 步 ，FastRunner 就 会 走 2p 步 。 因 此 ， 当 SlowRunner 走 了 k 步 进 
入 环 路 部 分 时 ，FastRunner 已 走 了 总 共 2k 步 ， 进 入 环 路 部 分 已 有 2k - k 步 或 k 步 。 由 于 k 可 能 比 环 
路 长 度 大 得 多 ， 实 际 上 我 们 应 该 将 它 写 作 mod(k，LOOP_SIZE) 步 ， 并 用 K 表 示 。 

对 于 之 后 的 每 一 步 ，FastRunner 和 SlowRunner 之 间 不 是 走 远 一 步 就 是 更 近 一 步 ， 具 体 要 看 
观察 的 角度 。 也 就 是 说 ， 因 为 两 者 处 于 圆圈 中 ， 当 A 以 远离 B 的 方向 走出 q 步 时 ， 同 时 也 是 向 B 更 
近 了 q 步 。 综 上 ， 我 们 得 出 以 下 几 点 : 

(1) SlowRunner 处 于 环 路 中 的 8 步 位 置 ; 

(2) FastRunner 处 于 环 路 中 的 K 步 位 置 ; 

(3) SlowRunner 落 后 于 FastRunner， 相 距 K 步 ; 

(4) FastRunner 落 后 于 SlowRunner， 相 距 LOOP_SIZE - K 步 ; 

(5) 每 过 一 个 单位 时 间 ，FastRunner 就 会 更 接近 SlowRunner 一 步 。 

那么 ， 两 个 结 点 什么 时 候 相 遇 ? 若 FastRunner 落 后 于 SlowRunner， 相 距 LOOP_SIZE - K 步 ， 
并 且 每 经 过 一 个 单位 时 间 ，FastRunner 就 走 近 SlowRunner 一 步 ， 那 么 ， 两 者 将 在 LOOP_SIZE -kK 
步 之 后 相遇 。 此 时 ， 两 者 与 环 路 起 始 处 相距 K 步 ， 我 们 将 这 个 位 置 称 为 ColLlisionspot。 


第 3 部 分 : 如 何 找到 环 路 起 始 处 ? 

现在 我 们 知道 Collisionspot 与 环 路 起 始 处 相距 K 个 结 点 。 由 于 K = mod(k，LOOP_STZE) (或 
者 换 句 话说 ，k = Kk + M * LOOP_STZE， 其 中 为 任意 整数 )， 也 可 以 说 ，Collisionspot 与 环 路 
起 始 处 相距 k 个 结 点 。 例 如 ， 若 有 个 环 路 长 度 为 $ 个 结 点 ， 有 个 结 点 N 处 于 距离 环 路 起 始 处 2 个 结 
点 的 地 方 ， 我 们 也 可 以 换个 说 法 : 这 个 结 点 处 于 距离 环 路 起 始 处 7 个 、12 个 甚至 397 个 结 点 。 











































































































图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 


9.2 链表 127 











EE 《| 


n1 和 n2 将 在 此 相遇 ， 
与 环 路 起 始 处 相距 3 个 结 点 一 一 一 > 





至 此 ，Collisionspot 和 LinkedListHead 与 环 路 起 始 处 均 相 距 k 个 结 点 。 

现在 ， 若 用 一 个 指针 指向 Collisionspot ， 用 男 一 个 指针 指向 LinkedListHead， 两 者 与 
Loopstart 均 相距 k 个 结 点 。 以 同样 的 速度 移动 ， 这 两 个 指针 会 再 次 碰 在 一 起 一 一 这 次 是 在 k 步 之 
后 ， 此 时 两 个 指针 都 指向 Loopstart， 然 后 只 需 返 回 该 结 点 即 可 。 


第 4 部 分 : 将 全 部 整合 在 一 起 

总 结 一 下 ，FastpPointer 的 移动 速度 是 SlowPointer 的 两 倍 。 当 SlowPointer 走 了 k 个 结 点 进 
人 环 路 时 ，FastpPointer 已 进入 链表 环 路 k 个 结 点 。 也 就 是 说 FastPointer 和 SlowPointer 相 距 
LOOP_SIZE - k 个 结 点 。 

接 下 来 ， 若 slowPointer 每 走 一 个 结 点 ，FastPointer 就 走 两 个 结 点 ， 每 走 一 次 ， 两 者 的 距 
离 就 会 更 近 一 个 结 点 。 因 此 ， 在 走 了 LOOP_SIZE - k 次 后 ， 它 们 就 会 碰 在 一 起 。 这 时 两 者 距离 环 
路 起 始 处 有 Kk 个 结 点 。 

链表 首部 与 环 路 起 始 处 也 相距 k 个 结 点 。 因 此 ， 若 其 中 一 个 指针 保持 不 变 ， 男 一 个 指针 指 问 
链表 首部 ， 则 两 个 指针 就 会 在 环 路 起 始 处 相 会 。 

根据 第 1、2、3 部 分 ， 就 能 直接 导出 下 面 的 算法 。 

(1) 创建 两 个 指针 : FastPointer 和 SlowPointer。 

(2) SlowPointer 每 走 一 步 ，FastPointer 就 走 两 步 。 

(3) 两 者 碰 在 一 起 时 ， 将 SlowPointer 指 向 LinkedListHead，FastPointer 保 持 不 变 。 

(4) 以 相同 速度 移动 SlowPointer 和 FastPointer， 一 次 一 步 ， 然 后 返回 新 的 碰撞 处 。 

下 面 是 该 算法 的 实现 代码 。 

1 LinkedListNode FindBeginning(LinkedListNode head) { 


LinkedListNode slow = head; 
LinkedListNode fast = head; 


















































* 位 置 */ 
while (fast != null && fast.next != null) { 
Slow = slow.next; 


2 
3 
4 
5 /* 找 出 碰撞 处 ， 将 处 于 链表 中 LOOP_SIZE - k 步 的 
6 
7 
8 


9 fast = fast.next.next; 

16 if (slow == fast) { // 碰撞 
11 break; 

可 4 } 
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15 /* 错误 检查 ， 没 有 碰撞 处 ， 也 即 没 有 环 路 */ 
16 if (fast == null || fast.next == null) { 
17 return null; 

yg 


26 /* 将 Slow 指向 首部 ，fast 指 向 碰撞 处 ， 两 者 
21 * 距离 环 路 起 始 处 K 步 ， 若 两 者 以 相同 速度 移动 ， 
22 * 则 必定 会 在 环 路 起 始 处 碰 在 一 起 */ 

23 slow = head; 

24 while (slow != fast) { 


25 slow = slow.next; 

26 fast = fast.next; 

27 } 

28 

29 /* 至 此 两 者 均 指 向 环 路 起 始 处 */ 
36 return fast; 

31 } 


2.7 ”编写 一 个 图 数 ， 检 查 链表 是 否 为 回 文 。( 第 49 页 ) 


解法 
要 解决 这 个 问题 ， 可 以 将 回 文 (palindrome ) 定义 为 6 -> 1 -> 2 -> 1 -> 6。 显 然 ， 知 链 
表 是 回 文 ， 不 管 正 着 看 还 是 反 着 看 ， 都 是 一 样 的 。 由 此 可 以 得 出 第 一 种 解法 。 


解法 1: 反 转 并 比较 

第 一 种 解法 是 反 转 整个 链表 , 然后 比较 反 转 链表 和 原始 链表 。 知 两 者 相同 , 则 该 链表 为 回 文 。 

注意 , 在 比较 原始 链表 和 反 转 链表 时 ， 其 实 只 需 比较 链表 的 前 半 部 分 。 若 原始 链表 和 反 转 链 
表 的 前 半 部 分 相同 ， 那 么 ， 两 者 的 后 半 部 分 肯定 相同 。 


解法 2: 迭代 法 

要 想 探测 链表 的 前 半 部 分 是 否 为 后 半 部 分 反 转 而 成 ， 该 怎么 做 呢 ? 只 需 将 链表 前 半 部 分 反 
转 ， 可 以 利用 栈 来 实现 。 

我 们 需要 将 前 半 部 分 结 点 人 栈 。 根 据 链表 长 度 已 知 与 否 ， 人 栈 有 两 种 方式 。 

若 链 表 长 度 已 知 ,可 以 用 标准 for 迭 代 访 问 前 半 部 分 结 点 , 将 每 个 结 点 入 栈 。 当 然 , 要 小 心 处 
理 链表 长 度 为 奇数 的 情况 。 

若 链 表 长 度 未 知 ， 可 以 利用 本 章 开头 描述 的 快慢 runner 方 法 迭代 访问 链表 。 在 迭代 循环 的 每 
一 步 ， 将 慢 速 runner 的 数据 入 栈 。 在 快速 runner 抵 达 链 表 尾 部 时 ， 慢 速 runner 刚 好 位 于 链表 中 间 位 
置 。 至 此 ， 栈 里 就 存放 了 链表 前 半 部 分 的 所 有 结 点 ， 不 过 顺序 是 相反 的 。 

接 下 来 ,我们 只 需 迭 代 访 问 链表 余下 结 点 。 每 次 迭代 时 ， 比 较 当 前 结 点 和 栈 顶 元 素 ， 若 完成 
迭代 时 比较 结果 完全 相同 ， 则 该 链表 是 回 文 序 列 。 


1 boolean isPalindrome(LinkedListNode head) { 
之 LinkedListNode fast = head; 
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3 LinkedListNode slow = head; 

4 

5 Stack<Integer> stack = new Stack<Integer>(); 

6 

7 /* 将 链表 前 半 部 分 元 素 入 栈 。 当 快速 runner (移动 速度 为 
8 * 慢 速 runner 的 两 倍 ) 到 达 链 表 尾 部 时 ， 则 慢 速 runner 已 
9 *# 处 在 链表 中 间 位 置 */ 

16 while (fast != null && fast.next != null) { 

1 stack.push(slow.data); 

12 slow = slow.next; 

13 fast = fast.next.next; 

14 } 

15 


16 /* 链表 有 坷 数 个 元 素 ， 跳 过 中 间 元 素 */ 
17 if (fast != null) { 


18 slow = slow.next; 

19 } 

20 

21 while (slow != null) { 

22 int top = stack.pop().intValue(); 
23 

24 /* 两 者 不 相同 ， 则 该 链表 不 是 回 文 序列 */ 
25 if (top != slow.data) { 

26 return false; 

27 } 

28 slow = slow.next; 

29 } 

30 return true; 

31 } 


解法 3: 递归 法 

首先 ， 简 要 介绍 下 面 的 解法 用 到 的 记号 : 用 记号 九 表 示 结 点 时 ， 变 量 K 指 示 结 点 数据 的 值 ， 
而 x( 取 f 或 b ) 指示 该 结 点 是 值 为 K 的 前 方 结 点 还 是 后 方 结 点 。 例 如 ， 在 下 面 的 链表 中 ， 结 点 3b 
指 的 是 值 为 3 的 第 二 个 (b 一 back， 即 后 方 ) 结 点 。 

接 下 来 ， 跟 许多 链表 问题 一 样 ， 可 以 用 递归 法 解决 这 个 问题 。 我 们 靠 直觉 可 能 就 会 想到 要 比 
较 元 素 0 和 元 素 n， 元 素 1 和 元 素 n-1， 元 素 2 和 元 素 n-2， 等 等 ， 直 至 中 间 元 素 。 

例如 : 

e(1(2(3)2)1)8 

为 了 运用 这 种 方法 ,首先 必须 知道 什么 时 候 到 达 中 间 元 素 , 这 也 形成 了 递归 的 终止 条 件 。 每 
次 递归 调用 传人 length - 2 为 长 度 ， 当 长 度 等 于 0 或 1 时 ， 表 明 当 前 已 处 于 链表 中 间 位 置 。 




















recurse(Node n, int length) { 
if (length == 6 || length == 1) { 
return [something]; // 中 间 


1 
2 
3 
4 } 
5 recurse(n.next, length - 2); 
6 
7 
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这 个 方法 构成 了 isPalindrome 方 法 的 轮廓 ， 而 该 算法 的 实质 则 是 比较 结 点 1 和 结 点 n - i， 


检查 链表 是 否 为 回 文 序列 。 具 体 该 怎么 做 呢 ? 


v1 = isPalindrome: list =06(1(2(3)2)1) 6. length=7 


仔细 分 析 下 面 的 调用 栈 : 

二 

2 v2 = isPalindrome: list = 1 
3 v3 = isPalindrome: list = 
4 v4 = isPalindrome: list 
5 returns v3 

6 returns v2 

ya returns v1 

8 returns ? 





( 
2 


\ 一 WU 一 


人 


9. length = 5 





在 上 面 的 调用 栈 中 , 每 次 调用 都 会 比较 其 首 结 点 和 链表 后 半 部 分 对 应 结 点 , 检查 链表 是 否 为 
回 文 序列 o 即 : 











引 ) 





口 第 1 行 需要 比较 结 点 ef 和 结 点 8b 
口 第 2 行 需 要 比较 结 点 1f 和 结 点 1b; 
口 第 3 行 需 要 比较 结 点 2f 和 结 点 2b 
口 第 4 行 需要 比较 结 点 3f 和 结 点 3b。 
者 将 上 面 的 栈 倒 过 来 ， 按 如 下 顺序 传 回 
D 第 4 行 发 现 传人 结 点 为 中 间 结 点 
因此 head.next 为 结 点 2b 

口 第 3 行 比较 首部 即 结 点 2f 和 returned_node ( 上 次 递归 调用 返回 的 值 ) 即 结 点 2b。 若 两 个 


人 


» 


; 





结 点 


Hi, 








我 们 只 需 : 


( 因为 length = 1 )， 传 回 head.next。 其 中 head 为 结 点 3 ， 


点 的 值 相 等 ， 则 传 回 结 点 lb 的 引用 ( returned_node .next ); 








口 第 2 行 比 较 首 部 ( 结 点 1f ) 和 returned_node ( 结 点 1b )。 若 两 个 结 点 的 值 相等 ， 则 传 回 结 
点 9b 的 引用 (或 returned_node.next ); 
口 第 1 行 比较 首部 ( 结 点 ef ) 和 returned_node( 结 点 gb )。 若 两 个 结 点 的 值 相等 , 则 返回 ture。 


归纳 一 下 ， 每 次 调用 都 会 比较 其 首部 和 returned_node， 然 后 回 传 returned_node.next。 
最 终 每 个 结 点 i 都 会 与 结 点 n - i 进行 比较 。 只 要 有 任意 一 对 结 点 的 值 不 相等 ， 就 立即 返回 false， 
调用 栈 的 上 一 级 调用 都 会 检查 这 个 布尔 值 。 
但 是 等 等 , 你 可 能 会 问 , 我 们 一 会 儿 说 要 返回 一 个 布尔 值 , 一 会 儿 说 要 返回 一 个 结 点 ?到 底 





返回 什么 ? 


两 个 都 要 返回 。 我 们 创建 了 一 个 包含 布尔 值 和 结 点 两 个 成 员 的 简单 类 , 调用 时 只 需 返 回 该 类 


的 实例 。 


1 
2 
3 
4 


class Result { 


} 


public LinkedListNode node; 
public boolean result; 


下 面 举例 说 明示 例 链表 每 次 递归 调用 的 参数 和 返回 值 。 


1 
2 





isPalindrome: list =6(1(2(3(4)3)2)1)686. len=9 
isPalindrome: list =1(2(3(4)3)2)1)0e. len=7 
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isPalindrome: list =2 (3(4)3)2)1)6. len=5 
isPalindrome: list =3(4)3)2)1)0e. len=3 
isPalindrome: list =4)3)2)1)0e. len=1 


returns node 2b, true 
returns node 1b, true 
returns node 68b, true 
16 returns nobe 8b, true 


至 此 ， 这 段 代码 实现 起 来 很 简单 ， 只 需 填 入 细节 即 可 。 


3 
4 
5 
6 returns node 3b, true 
7 
8 
9 


1 Result isPalindromeRecurse(LinkedListNode head, int length) { 
2 if (head == null || length == 6) { 

3 return new Result(null, true); 

4 } else if (length == 1) { 

5 return new Result(head.next, true); 
6 } else if (length == 2) { 

7 return new Result(head.next.next, 

8 head.data == head.next.data); 
9 


} 
16 Result res = ispalindromeRecurse(head.next, length - 2); 
和 型 if (!lres.result || res.node == null) { 
12 return res; 
13 } else { 
14 res.result = head.data == res.node.data; 
15 res.node = res.node.next; 
16 return res; 
17 } 
18 } 
19 


26 boolean isPalindrome(LinkedListNode head) { 

21 Result p = ispalindromeRecurse(head, listSize(head)); 
22 return p.result; 

23 } 


有 些 人 可 能 会 有 疑问 , 为 什么 要 这 么 费力 专门 创建 一 个 Result 类 , 有 没有 更 好 的 办 法 ?还 真 
没有 ， 至 少 用 Java 实 现 的 话 没有 。 
然而 ， 若 用 C 或 C++ 实现 的 话 ， 我 们 可 以 传人 一 个 指针 的 指针 。 


1 bool isPalindromeRecurse(Node head, int length, Node** next) { 
2 


3 } 
代码 不 太 好 看 ， 但 行 之 有 效 。 


9.3 栈 与 队列 





3.1 描述 如 何 只 用 一 个 数组 来 实现 三 个 栈 。( 第 50 页) 





解法 
和 许多 问题 _ 样 ， 这 个 问题 的 解法 基本 上 取决 于 你 机 对 楼 支持 到 什么 程度 者 每 个 本 分 本 加 
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132 第 9 章 解 题 技巧 
定 大 小 的 空间 ， 就 能 满足 需要 ， 那么 照 做 便 是 。 不 过 ,这么 做 的 话 ， 有 可 能 其 WE 
够 用 了 ， 而 同时 其 他 的 栈 却 几乎 是 空 的 。 另 一 种 做 法 是 弹性 处 理 栈 的 空间 分 配 ， 但 这 么 一 来 ， 


这 个 问题 的 复杂 度 又 会 大 大 增加 。 


方法 1: 固定 分 割 
我 们 可 以 将 整 全 人 半生 民居 向 种 各 上 网 宇 辣 时 全 大 证 且 


表示 包含 





口 
下 


DovamA 上 wmwN 情 


Pp 
卢 QQ 


PppPpApPpPp 
saw 人 wI 


WwwWwWwwwNDPDNDPDINDPDINPDNPPINPPDNPNNDP 上 情 
mwbPPGoDoo、vamFcwPQONO oo 


Wy 
| 





含 端点 ,“(” 表 示 不 包含 端点 。 
口 栈 1， n/3)。 


口 栈 2, 使 用 [n/3，2n/3)。 


栈 3， 使 用 [2n/3，n)。 
面 是 该 解法 的 实现 代码 。 
int stackSize = 166; 


int[] buffer = new int [stackSize * 3]; 
int[] stackPointer = {-1，-1，-1}; // 用 于 追踪 栈 顶 元 素 的 指针 








void push(int stackNum, int value) throws Exception { 

/* 检查 有 无 空闲 空间 */ 

if (stackPointer[stackNum] + 1 >= stackSize) { // 最 后 一 个 元 素 
throw new Exception(“Out of space.”); 

} 

/* 栈 指 针 自 增 ， 然 后 更 新 栈 顶 元 素 的 值 */ 

stackPointer[stackNum]++; 

buffer[absTopOfstack(stackNum)] = value; 


小 


int pop(int stackNum) throws Exception { 
if (stackPointer[stackNum] == -1) { 
throw new Exception(“Trying to pop an empty stack.”); 
} 
int value = buffer[absTopOfStack(stackNum)]; // 获取 栈 顶 元 素 的 值 
buffer[absTopOfStack(stackNum)] = 6; // 清 零 指定 索引 元 素 的 值 
stackPointer[stackNum]--; // 指针 自 减 
return value; 


} 


int peek(int stackNum) { 
int index = absTopOfSstack(stackNum) ; 
return buffer[index]; 


} 


boolean isEmpty(int stackNum) { 
return stackPointer[stackNum] == -1; 


} 


/* 返回 栈 “stackNum2 栈 顶 元 素 的 索引 ， 绝 对 量 */ 
int absTopOfStack(int stackNum) { 
return stackNum * stackSize + StackPointer[stackNum]; 


} 
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如 果 知 道 与 这 些 栈 的 使 用 情况 相关 的 更 多 信息 ， 就 可 以 对 上 面 的 算法 做 相应 的 改进 。 例 如 ， 
若 预 估 Stack 1 的 元 素 比 stack 2 多 很 多 ， 那 么 ， 就 可 以 给 stack 1 多 分 配 一 点 空间 ， 给 stack 2 
少 分 配 一 些 空间 。 


方法 2: 弹性 分 割 
第 二 种 做 法 是 允许 栈 块 的 大 小 灵活 可 变 。 当 一 个 栈 的 元 素 个 数 超出 其 初始 容量 时 , 就 将 这 个 
栈 扩容 至 许可 的 容量 ， 必 要 时 还 要 搬移 元 素 。 
此 外 ， 我 们 会 将 数组 设计 成 环 状 的 ， 最 后 一 个 栈 可 能 从 数组 末尾 开始 ， 环 绕 到 数组 开头 。 
请 注意 , 这 种 解法 的 代码 远 比 面试 中 常见 的 要 复杂 得 多 。 你 可 以 试 着 提供 伪 码 , 或 是 其 中 某 
几 部 分 的 代码 ， 但 要 完整 实现 的 话 ， 难 度 就 有 点 大 了 。 
/* StackData 是 个 简单 的 类 ， 存 放 每 个 栈 相 关 的 数据 ， 
* 但 并 未 存放 栈 的 实际 元 素 */ 
public class StackData { 


1 
2 
3 
4 public int start; 
5 public int pointer; 
6 
7 
8 
9 

















public int size = 0@; 
public int capacity; 
public StackData(int _start, int _capacity) { 


start = _start; 
16 pointer = _start - 1; 
11 capacity = _capacity; 
12 } 
13 
14 public boolean isWithinSstack(int index, int total size) { 
15 /* 注意 : 如 果 栈 回 绕 了 ， 首 部 ( 右 侧 ) 回 绕 到 
16 * 左边 */ 
17 if (start <= index && index < start + capacity) { 
18 // 不 回 绕 ， 或 回 绕 时 的 “首部 ”( 右 侧 ) 
19 return true; 
26 } else if (start + capacity > total_size && 
21 index < (start + capacity) % total size) { 
22 // 回 绕 时 的 尾部 ( 左 侧 ) 
23 return true; 
24 } 
25 return false; 
26 } 
27 } 
28 
29 public class QuestionB { 
36 static int number_of_stacks = 3; 
31 static int default size = 4; 
32 static int total_ size = default size * number of_stacks; 
33 static StackData [] stacks = {new StackData(60, default size), 
34 new StackData(default _ size, default _ size), 
35 new StackData(default size * 2, default size)}; 
36 static int [] buffer = new int [total sizel]; 
37 
38 public static void main(String [] args) throws Exception { 
39 push(86，16); 
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46 push(1，26); 
41 push(2，36) 1; 
42 int v = pop(@); 
43 A 
44 } 
45 
46 public static int numberOfElements() { 
47 return stacks[8].size + stacks[1].size + stacks[2].size; 
48 } 
49 
56 public static int nextElement(int index) { 
51 if (index + 1 == total_size) return ©; 
352 else return index + 1; 
53 } 
54 
55 public static int previousElement(int index) { 
56 if (index == 9) return total size - 1; 
57 else return index - 1; 
58 } 
59 
66 public static void shift(int stackNum) { 
61 StackData stack = stacks[stackNum]; 
62 if (stack.size >= stack.capacity) { 
63 int nextStack = (stackNum + 1) % number_of_stacks; 
64 shift(nextSstack); // 腾 出 若干 空间 
65 stack.capacity++; 
66 } 
67 
68 // 以 相反 顺序 搬移 元 素 
69 for (int i = (stack.start + stack.capacity - 1) % total size; 
76 stack.iswWithinSstack(i, total_ size); 
7 i = previousElement(i)) { 
72 buffer[i] = buffer[previousElement(i)]; 
73 } 
74 
75 buffer[stack.start] = 8; 
76 stack.start = nextElement(stack.start); // 移动 栈 的 起 始 位 置 
77 stack.pointer = nextElement(stack.pointer); // 移动 指针 
78 stack.capacity--; // 恢复 到 原先 的 容量 
79 } 
80 
81 /* 搬移 到 其 他 栈 上 ， 以 扩大 栈 的 容量 */ 
82 public static void expand(int stackNum) { 
83 shift((stackNum + 1) % number_of_stacks); 
84 stacks[stackNum] .capacity++; 
85 } 
86 
87 public static void push(int stackNum, int value) 
88 throws Exception { 
89 StackData stack = stacks[stackNum]; 
96 /* 检查 空间 够 不 够 */ 
91 if (stack.size >= stack.capacity) { 
92 if (numberOfElements() >= total_size) { // 全 部 都 满 了 
93 throw new Exception(“Out of space.”); 
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94 } else { // 只 是 需要 搬移 元 素 

95 expand(stackNum); 

96 } 

97 } 

98 /* 找 出 顶端 元 素 在 数组 中 的 索引 值 ， 并 加 1， 

99 * 并 增加 栈 指针 */ 

166 stack.sizett+; 

161 stack.pointer = nextElement(stack.pointer); 
162 buffer[stack.pointer] = value; 

163 } 

164 

165 public static int pop(int stackNum) throws Exception { 
166 StackData stack = stacks[stackNum] ; 

167 if (stack.size == 6) { 

168 throw new Exception(“Trying to pop an empty stack.”); 
169 } 

116 int value = buffer[stack.pointer]; 

111 buffer[stack.pointer] = 0; 

112 stack.pointer = previousElement(stack.pointer); 
113 stack.size--; 

114 return value; 

115 } 

116 

117 public static int peek(int stackNum) { 

118 StackData stack = stacks[stackNum]; 

119 return buffer[stack.pointer]; 

120 } 

121 

122 public static boolean isEmpty(int stackNum) { 
123 StackData stack = stacks[stackNum]; 

124 return stack.size == 0@; 

125 } 

126 } 


遇 到 类 似 的 问题 ， 应 该 力求 编写 清晰 、 可 维护 的 代码 ， 这 很 重要 。 你 应 该 引入 额外 的 类 ， 比 
如 这 里 使 用 了 stackData， 并 将 大 块 代码 独立 为 单独 的 方法 。 当 然 ， 这 个 建议 同样 适用 于 真正 的 
软件 开发 。 


3.2 请 设计 一 个 栈 ， 除 pop 与 push 方 法 ， 还 支持 min 方 法 ， 可 返回 栈 元 素 中 的 最 小 值 。push、 
pop 和 min 三 个 方法 的 时 间 复 杂 度 必须 为 O(1)。( 第 50 页 ) 


解法 

既然 是 最 小 值 ， 就 不 会 经 常 变动 ， 只 有 在 更 小 的 元 素 加 入 时 ， 才 会 改变 。 

一 种 解法 是 在 stack 类 里 添加 一 个 int 型 的 minValue。 当 minValue 出 栈 时 ， 我 们 会 搜索 整个 
栈 ， 找 出 新 的 最 小 值 。 可 惜 ， 这 不 符合 人 栈 和 出 栈 操作 时 间 为 0(1) 的 要 求 。 

为 进一步 理解 这 个 问题 ， 下 面 用 一 个 简短 的 例子 加 以 说 明 : 

push(5); // 栈 为 {5}， 最 小 值 为 5 


push(6); // 栈 为 {6，5}， 最 小 值 为 5 
push(3); // 栈 为 {3，6，5}， 最 小 值 为 3 
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push(7); // 栈 为 {7， 3， 6， 5 和 最 小 值 为 3 
pop(); // 弹出 7， 栈 为 {3，6，5},， 最 小 值 为 3 
pop(); // 弹出 3， 栈 为 {6，5},， 最 小 值 为 5 


注意 观察 ， 当 栈 回 到 之 前 的 状态 ( {6, 5} ) 时 ， 最 小 值 也 回 到 之 前 的 状态 (5 )， 这 就 导出 了 
我 们 的 第 二 种 解法 。 

只 要 记 下 每 种 状态 的 最 小 值 , 我 们 就 能 轻易 获取 最 小 值 。 实 现 很 简单 ， 每 个 结 点 记录 当前 最 
小 值 即 可 。 这 么 一 来 ， 要 找到 min， 直 接 查 看 栈 顶 元 素 就 能 得 到 最 小 值 。 

当 一 个 元 素 人 栈 时 ， 该 元 素 会 记 下 当前 最 小 值 ， 将 min 记 录 在 自身 数据 结构 的 min 成 员 中 。 




















1 public class StackWithMin extends Stack<NodeWithMin> { 
2 public void push(int value) { 

3 int newMin = Math.min(value, min()); 

4 super.push(new NodeWithMin(value, newMin)); 
5 } 

6 

- public int min() { 

8 if (this.isEmpty()) { 

9 return Integer.MAX_VALUE; // 错误 值 

16 } else { 

1 return peek() .min; 

12 } 

13 } 

14 } 

15 


16 class NodeWithMin { 
17 public int value; 


18 public int min; 

19 public NodeWithMin(int v, int min){ 
26 value = Vi 

21 this.min = min; 

22 } 

23 } 























但 是 ， 这 种 做 法 有 个 缺点 : 当 栈 很 大 时 ， 每 个 元 素 都 要 记录 min， 就 会 浪费 大 量 空 间 。 还 有 
没有 更 好 的 做 法 ? 
利用 额外 的 栈 来 记录 这 些 min， 我 们 也 许可 以 比 之 前 做 得 更 好 一 点 。 


1 public class StackWithMin2 extends Stack<Integer> { 
2 Stack<Integer> s2; 

3 public StackwithMin2() { 

4 s2 = new Stack<Integer>(); 
5 } 

6 

7 public void push(int value){ 
8 if (value <= min()) { 

9 s2.push(value); 

16 } 

4 于 super.push(value); 

12 } 

13 


14 public Integer pop() { 
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15 int value = super.pop(); 
16 if (value == min()) { 

17 s2.pop(); 

18 } 

19 return value; 

26 } 

21 

22 public int min() { 

23 if (s2.isEmpty()) { 

24 return Integer.MAX_VALUE; 
25 } else { 

26 return s2.peek(); 

27 } 

28 } 

29 } 


为 什么 这 么 做 可 以 节省 空间 ?” 假设 有 个 很 大 的 栈 , 而 第 一 个 元 素 刚 好 是 最 小 值 。 对 于 第 一 种 
解法 ,我 们 需要 记录 n 个 整数 ， 其 中 n 为 栈 的 大 小 。 不 过 ,对 于 第 二 种 解法 ,我 们 只 需 存 储 几 项 数 
据 : 第 二 个 栈 ( 只 有 一 个 元 素 )， 以 及 栈 本 身 数据 结构 的 若干 成 员 。 


3.3 ”设想 有 一 堆 盘 子 ， 堆 太 高 可 能 会 倒 下 来 。 因 此 ， 在 现实 生活 中 ， 盘 子 堆 到 一 定 高 度 
时 ， 我 们 就 会 另外 堆 一 堆 盘 子 。 请 实现 数据 结构 SetofSstacks ， 模 拟 这 种 行为 。setOfstacks 应 
该 由 多 个 栈 组 成 ， 并 且 在 前 一 个 栈 填 满 时 新 建 一 个 栈 。 此 外 ，setofstacks.push() 和 
Setofstacks.pop() 应 该 与 普通 栈 的 操作 方法 相同 〈 也 就 是 说 ，pop() 返 回 的 值 ， 应 该 跟 只 
个 栈 时 的 情况 一 样 )。 

进 阶 

实现 一 个 popAt(int index) 方 法 ， 根 据 指定 的 子 栈 ， 执 行 pop 操 作 。( 第 51 页 ) 

解法 

在 这 个 问题 中 ， 根 据 题 意 ， 数 据 结构 应 该 类 似 如 下 : 


1 class SetofStacks { 


2 ArrayList<Stack> stacks = new ArrayList<Stack>(); 
3 public void push(int v) { ... } 

4 public int pop() { ... } 

5 } 





其 中 push() 的 行为 必须 跟 单一 栈 的 一 样 ， 这 就 意味 着 push() 要 对 栈 数 组 的 最 后 一 个 栈 调 用 
push()。 不 过 , 这 里 处 理 起 来 必须 小 心 一 点 : 若 最 后 一 个 栈 被 填 满 ， 就 需 新 建 一 个 栈 。 实 现代 码 
大 致 如 下 : 





1 public void push(int v) { 

2 Stack last = getLastStack(); 

3 if (last != null && !last.isFull()) { // 添加 到 最 后 一 个 栈 中 
4 last.push(v); 

5 } else { // 必须 新 建 一 个 栈 

6 Stack stack = new Stack(capacity ) ; 

7 stack.push(v); 

8 stacks.add(stack); 
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16 } 
那么 ，pop() 该 怎么 做 ? 它 的 行为 与 push() 类 似 ， 也 就 是 说 ， 应 该 操作 最 后 一 个 栈 。 知 最 后 
一 个 栈 为 空 〈 执行 出 栈 操作 后 )， 就 必须 从 栈 数 组 中 移 除 这 个 栈 。 





1 public int pop() { 

2 Stack last = getLastStack(); 

3 int v = last.pop(); 

4 if (last.size == 6) stacks.remove(stacks.size() - 1); 
3 return v; 

6 


} 


进 阶 ， 实现 popAt(int index) 

这 个 实现 起 来 有 点 琼 手 ， 不 过 ， 我 们 可 以 设想 一 个 “ 推 人 ”动作 。 从 栈 1 弹 出 元 素 时 ,我们 
需要 移出 栈 2 的 栈 底 元 素 ， 并 将 其 推 到 栈 1 中 。 随 后 ， 将 栈 3 的 栈 底 元 素 推 人 栈 2， 将 栈 4 的 栈 底 元 
素 推 人 栈 3 ， 等 等 。 

你 可 能 会 指出 ， 何 必 执 行 “ 推 人 ”操作 ， 有 些 栈 不 填 满 不 是 也 挺 好 的 。 而 且 ， 这 还 会 改善 时 
间 复 杂 度 〈 元素 很 多 时 尤其 明显 )， 但 是 ， 若 之 后 有 人 假定 所 有 的 栈 (最 后 一 个 栈 除外 ) 都 是 填 
满 的 ， 就 可 能 会 让 我 们 陷 人 环 手 的 境地 。 这 个 问题 并 没有 “标准 答案 ”， 你 应 该 跟 面试 官 讨论 各 
种 做 法 的 优 劣 。 












































1 public class SetofStacks { 

2 ArrayList<Stack> stacks = new ArrayList<Stack>(); 
3 public int capacity; 

4 public SetofStacks(int capacity) { 

5 this.capacity = capacity; 

6 
7 
8 


} 

public Stack getLastStack() { 
9 if (stacks.size() == 6) return null; 
16 return stacks.get(stacks.size() - 1); 
1 } 
12 


13 public void push(int v) { /* 参看 之 前 的 代码 */ } 
14 public int pop() { /* 参看 之 前 的 代码 */ } 
15 public boolean isEmpty() { 


16 Stack last = getLastStack(); 

17 return last == null || last.isEmpty(); 

18 } 

19 

26 public int popAt(int index) { 

21 return leftShift(index, true); 

22 } 

23 

24 public int leftShift(int index, boolean removeTop) { 
25 Stack stack = stacks.get(index); 

26 int removed_item; 

27 if (removeTop) removed item = stack.pop(); 
28 else removed item = stack.removeBottom(); 
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29 if (stack.isEmpty()) { 

36 stacks.remove(index); 

31 } else if (stacks.size() > index + 1) { 
32 int v = leftShift(index + 1, false); 
33 stack.push(v); 

34 } 

35 return removed_item; 

36 } 

37 } 

38 

39 public class Stack { 

46 private int capacity; 


41 public Node top, bottom; 
42 public int size = 0@; 


43 

44 public Stack(int capacity) { this.capacity = capacity; } 
45 public boolean isFull() { return capacity == size; } 
46 

47 public void join(Node above, Node below) { 
48 if (below != null) below.above = above; 
49 if (above != null) above.below = below; 
56 } 

51 

52 public boolean push(int v) { 

53 if (size >= capacity) return false; 

54 Size++; 

55 Node n = new Node(v); 

56 if (size == 1) bottom = nN; 

57 join(n, top); 

58 top = n; 

59 return true; 

66 } 

61 

62 public int pop() { 

63 Node t = top; 

64 top = top.below; 

65 Size--; 

66 return t.value; 

67 } 

68 

69 public boolean isEmpty() { 

70 return size == 0; 

71 } 

72 

73 public int removeBottom() { 

74 Node b = bottom; 

75 bottom = bottom.above; 

76 if (bottom != null) bottom.below = null; 
77 Size--; 

78 return b.value; 

79 } 














80 } 
这 个 问题 在 概念 上 并 不 是 很 难 , 但 要 完整 实现 需要 编写 大 量 代码 。 面 试 官 一 般 不 会 要 求 你 写 9 
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出 全 部 代码 。 

















解决 这 类 问题 ， 有 个 很 好 的 策略 ， 就 是 尽量 将 代码 分 离 出 来 ， 写 成 独立 的 方法 ， 比 如 popAt 


可 以 调用 的 leftshift。 这 样 一 来 ， 你 的 代码 就 会 更 加 清晰 ， 而 你 在 处 理 细节 之 前 ， 也 有 机 会 先 


铺设 好 代码 的 骨架 。 


3.4 在 经 典 问题 汉 诺 塔 中 ， 有 3 根 柱 子 及 N 个 不 同 大 小 的 穿孔 圆 盘 ， 盘 子 可 以 滑 入 任意 一 根 


柱子 。 一 开始 ， 所 有 盘子 自 底 向 上 从 大 到 小 依次 套 在 第 一 根 柱子 上 《〈 即 每 一 个 盘子 只 能 放 在 更 大 





的 盘子 上 面 )。 移 动 圆 盘 时 有 以 下 限制 : 
(1) 每 次 只 能 移动 一 个 盘子 ; 
(2) 盘子 只 能 从 柱子 顶端 滑 出 移 到 下 一 根 柱 子 ; 
(3) 盘子 只 能 苞 在 比 它 大 的 盘子 上 。 
请 运用 栈 ， 编 写 程序 将 所 有 盘子 从 第 一 根 柱子 移 到 最 后 一 根 柱子 。( 第 51 页 ) 


解法 
这 个 问题 看 起 来 很 适 


我 们 先 从 最 简单 的 例 


合 采用 基本 案例 构建 法 。 


子 7 = 1 开始 。 





当 n = 1 时 ， 能 否 将 盘子 1 从 柱 1 移 至 柱 3? 回答 是 肯定 的 。 








直接 将 盘子 1 从 柱 1 移 














至 柱 3。 








当 n = 2 时 ， 能 和 否 将 盘子 1 和 盘子 2 从 柱 1 移 至 柱 3? 可 以 。 

(1) 将 盘子 1 从 柱 1 移 至 柱 2。 

(2) 将 盘子 2 从 柱 1 移 至 柱 3。 

(3) 将 盘子 1 从 柱 2 移 至 柱 3。 

注意 ， 上 述 步骤 将 柱 2 用 作 缓 冲 区 ， 在 我 们 将 其 他 盘子 移 至 柱 3 时 ， 柱 2 会 暂 存 一 个 盘子 。 
当 n = 3 时 ， 能 否 将 盘子 1、2、3 从 柱 1 移 至 柱 3? 可 以 。 


(1) 从 上 面 可 知 ， 我 人 














么 做 了 ， 只 不 过 ， 这 里 是 将 这 两 个 盘子 移 至 柱 2。 


(2) 将 盘子 3 移 至 柱 3。 


(3) 将 盘子 1 、2 移 至 柱 3。 重 复 步 又 1 即 可 。 
当 n = 4 时 ， 能 否 将 盘子 1、2、3、4 从 柱 1 移 至 柱 3? 可 以 。 
(1) 将 盘子 1、2、3 移 至 柱 2。 具 体 做 法 参见 前 面 的 例子 。 


(2) 将 盘子 4 移 至 柱 3。 








(3) 将 盘子 1、2 、3 移 至 柱 3。 
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主意 ， 柱 2 和 柱 3 之 间 并 无 多 大 区 别 ， 只 是 叫 法 不 一 样 ， 实 则 是 等 价 的 。 把 柱 2 作 为 缓冲 ， 以 
将 盘子 移 至 柱 3， 相 比 把 柱 3 用 作 缓 冲 ， 以 将 盘子 移 至 柱 2， 并 无 区 别 。 

根据 上 述 做 法 ,很 自然 地 就 可 以 导出 递归 算法 。 在 每 一 部 分 ,我 们 都 会 执行 以 下 步骤 ,用 伪 
码 简 述 如 下 : 


1 moveDisks(int n, Tower origin, Tower destination, Tower buffer) { 
2 /* 终止 条 件 */ 

3 if (n <= 6) return; 

4 

5 /* 将 顶端 n - 1 个 盘子 从 origin 移 至 buffer， 

6 * 将 destination 用 作 缓 冲 区 。 */ 

也 moveDisks(n - 1, origin, buffer, destination); 
8 

9 /* 将 origin 顶 冶 的 盘子 移 至 destination */ 

16 moveTop(origin, destination); 

1 


12 /* 将 顶部 n - 1 个 盘子 从 buffer 移 至 destination， 

13 * 将 origin 用 作 缓 冲 区 。 */ 

14 moveDisks(n - 1, buffer, destination, origin); 
15 } 


下 面 的 代码 给 出 了 这 个 算法 更 详细 的 实现 ， 其 中 还 运用 了 面向 对 象 设计 思想 。 

















1 public static void main(String[] args) 
2 int n = 3; 

3 Tower[] towers = new Tower[n]; 

4 for (int i = 6;j i < 3; i++) { 

5 towers[i] = new Tower(i); 

6 } 

了 

8 for (int i =n- 1; i >= 6; 1i--) 1{ 
9 towers[6].add(i); 

16 } 

11 towers[8].moveDisks(n, towers[2], towers[1]); 
12 } 

13 


14 public class Tower { 
15 private Stack<Integer> disks; 


16 private int index; 

17 public Tower(int i) { 

18 disks = new Stack<Integer>(); 

19 index = i; 

26 } 

21 

22 public int index() { 

23 return index; 

24 } 

25 

26 public void add(int d) { 

27 if (!disks.isEmpty() && disks.peek() <= d) { 
28 System.out.println("EPror placing disk " + d); 
29 } else { 

36 disks.push(d); 
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34 public void moveTopTo(Tower 七 ) { 

35 int top = disks.pop(); 

36 t.add(top); 

37 System.out.println("Move disk " + top + " from " + index() + 
38 "to" +t.index()); 

39 } 


41 public void moveDisks(int n, Tower destination, Tower buffer) { 
42 if (n > 8) { 

43 moveDisks(n - 1, buffer, destination); 

44 moveTopTo(destination); 

45 buffer.moveDisks(n - 1, destination, this); 





严格 来 说 , 并 不 一 定 要 将 柱子 实现 为 独立 的 对 象 , 不 过 , 在 某 种 程度 上 ， 这 人 么 做 可 使 代码 更 
清晰 易 读 。 


3.5 ”实现 一 个 MyQueue 类 ， 该 类 用 两 个 栈 来 实现 一 个 队列 。( 第 51 页 ) 


解法 

队列 和 栈 的 主要 区 别 在 于 元 素 进 出 顺序 ( 先进 先 出 和 后 进 先 出 ), 因此 , 我 们 需要 修改 peek() 
和 pop()， 以 相反 顺序 执行 操作 。 我 们 可 以 利用 第 二 个 栈 反 转 元 素 的 次 序 (弹出 s1 的 元 素 ， 压 人 
s2 )。 在 这 种 实现 中 ， 每 当 执行 peek() 和 pop() 操 作 时 ， 就 要 将 s1 的 所 有 元 素 弹 出 ， 压 人 s2 中 ， 
然后 执行 peek/pop 操 作 ， 再 将 所 有 元 素 压 人 s1。 

上 述 做 法 也 是 可 行 的 ,但 若 连 续 执 行 两 次 pop/peek 操 作 , 那么 ， 所 有 元 素 都 要 移 来 移 去 , 重 
复 移动 ,这 毫 无 必要 。 我 们 可 以 延迟 元 素 的 移动 ， 即 让 元 素 一 直 留 在 s2 中 ， 只 有 必须 反 转 元 素 次 
序 时 才 移 动 元 素 。 

在 这 种 做 法 中 ，stackNewest 顶 端 为 最 新 元 素 ，stack01ldest 顶 端 为 最 旧 元 素 。 在 将 一 个 元 
素 出 列 时 , 我 们 希望 先 移 除 最 旧 元 素 , 因此 先 将 元 素 从 stack0ldest 将 元 素 出 列 。 若 stackOldest 
为 空 ， 则 将 stackNewest 中 的 所 有 元 素 以 相反 的 顺序 转移 到 stackoldest 中 。 如 要 插入 元 素 ， 就 
将 其 压 人 stackNewest， 因 为 最 新 元 素 位 于 它 的 顶端 。 

下 面 是 该 算法 的 实现 代码 。 











public class MyQueue<T> { 
Stack<T> stackNewest, stackOldest; 


public MyQueue() { 


1 

2 

3 

4 

5 stackNewest = new Stack<T>(); 
6 stackOldest = new Stack<T>(); 
yl 

8 

9 


} 


public int size() { 
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16 return stackNewest.size() + stackOldest.size(); 
11 } 

12 

13 public void add(T value) { 

14 /* 压 入 stackNewest， 最 新 元 素 始 终 位 于 它 

15 * 的 顶 问 */ 

16 stackNewest.push(value); 

17 } 

18 


19 /* 将 元 素 从 stackNewest 移 至 stackO1dest， 这 么 做 通常 是 
26 * 为 了 要 在 stackOldest 上 执行 操作 */ 
21 private void shiftSstacks() { 


22 if (stackOldest.isEmpty()) { 

23 while (!stackNewest.isEmpty()) { 

24 stackOldest.push(stackNewest.pop()); 
25 } 

26 } 

27 } 

28 

29 public T peek() { 

36 shiftstacks(); // 确保 stackOldest 含 有 当前 元 素 
31 return stackOldest.peek(); // 取 回 最 旧 元 素 
32 } 

33 

34 public T remove() { 

35 shiftstacks(); // 确保 stackOldest 含 有 当前 元 素 
36 return stackOldest.pop(); // 弹出 最 旧 元 素 

37 } 

38 } 


在 真正 的 面试 中 ,你 有 可 能 记 不 清 具 体 的 API 调 用 。 真 的 碰 到 这 种 情况 时 ， 也 不 必 太 紧张 。 
你 可 以 问 一 些小 细节 ， 多 数 面 试 官 都 不 会 为 难 你 。 他 们 更 关注 你 能 否 做 到 通盘 的 理解 问题 。 


3.6 ”编写 程序 ， 按 升序 对 栈 进行 排序 ( 即 最 大 元 素 位 于 栈 顶 )。 最 多 只 能 使 用 一 个 额外 的 
栈 存放 临时 数据 ， 但 不 得 将 元 素 复制 到 别 的 数据 结构 中 (如 数组 )。 该 栈 支 持 如 下 操作 : push、 
pop、peek 和 isEmpty。( 第 51 页 ) 


解法 

一 种 做 法 是 实现 初步 的 排序 算法 。 搜 索 整 个 栈 ， 找 出 最 小 元 素 ， 之 后 将 其 压 入 男 一 个 栈 。 然 
后 ， 在 剩余 元 素 中 找 出 最 小 的 ， 并 将 其 入 栈 。 这 种 做 法 实际 上 需要 三 个 栈 : s1 为 原先 的 栈 ，s2 
为 最 终 排 好 序 的 栈 ，s3 在 搜索 s1 时 用 作 缓 冲 区 。 要 在 s1 中 搜索 最 小 值 ， 我 们 需要 弹出 s1 的 元 素 ， 
将 它们 压 入 缓冲 区 s3。 

可 异 ， 我 们 只 能 使 用 一 个 额外 的 栈 。 有 没有 更 好 的 做 法 ?有 。 

我 们 不 需要 反复 搜索 最 小 值 ， 若 要 对 s1 排 序 ， 可 以 从 s1 逐 一 弹出 元 素 ， 然 后 按 顺 序 搬 入 s2 
中 。 具 体 怎么 做 呢 ? 

假设 有 如 下 两 个 栈 ， 其 中 s2 是 “排序 的 "，s1 则 是 未 排序 的 : 
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Ss1 s2 

12 
5 8 
16 3 
7 1 














从 si 中 弹出 5 时 ,我们 需要 在 s2 中 找 个 合适 的 位 置 插 入 这 个 数 。 在 这 个 例子 中 ， 正 确 位 置 是 
在 s2 元 素 3 之 上 。 怎 样 才能 将 5 插入 那个 位 置 呢 ? 我 们 可 以 先 从 si 中 弹出 5， 将 其 存放 在 临时 变量 
中 。 然 后 ， 将 12 和 8 移 至 s1 ( 从 s2 中 弹出 这 两 个 数 ， 并 将 它们 压 人 s1 中 )， 然 后 将 5 压 入 s2。 













































































步骤 1 步骤 2 步骤 3 

s1 s2 SI s2 
12 8 
8 -> 12 -> 

16 3 16 3 

7 1 7 1 

tmp 兰 号 tmp = 5 tmp 二 

注意 ，8 和 12 仍 在 s1 中 ,这 没关系 ! 对 于 这 两 个 数 , 我 们 可 以 像 处 理 5 那 样 重复 相关 步 又 ， 


轩 局 


次 弹出 s1 栈 顶 元 素 ， 将 其 放 人 s2 中 的 合适 位 置 。( 当然 ， 我 们 可 以 将 8 和 12 直 接 从 s2 移 至 s1， 
为 这 两 个 数 都 比 5 大 ， 这 些 元 素 的 “正确 位 置 ”就 是 放 在 5 之 上 。 我 们 不 需要 打 乱 s2 的 其 他 元 素 ， 
当 tmp 为 8 或 12 时 ， 下 面 代码 中 的 第 二 个 while 循 环 不 会 执行 。) 


1 public static Stack<Integer> sort(Stack<Integer> s) { 
2 Stack<Integer> r = new Stack<Integer>(); 





3 while (!s.isEmpty()) { 

4 int tmp = s.pop(); // 步 骤 1 

5 while (!Lr.isEmpty() && r.peek() > tmp) { // 步 骤 2 
6 s.push(r.pop()); 

7 } 

8 r.push(tmp); / /步骤 3 

9 

16 return r; 

11 } 





这 个 算法 的 时 间 复 杂 度 为 O0Y)， 空 间 复 杂 度 为 O(N)。 

如 果 人 允许 使 用 的 栈 数 量 不 限 ， 我 们 可 以 实现 修改 版 的 quicksort 或 mergesort。 

对 于 mergesort 解 法 ,我 们 可 以 再 创建 两 个 栈 , 并 将 这 个 栈 分 为 两 部 分 。 我 们 会 递归 排序 每 个 
栈 ,， 然 后 将 它们 归并 到 一 起 并 排 好 序 ， 放 回 原来 的 栈 中 。 注 意 ， 该 解法 要 求 每 层 递 归 都 创建 两 个 
额外 的 栈 。 

对 于 quicksort 解 法 ， 我 们 会 创建 两 个 额外 的 栈 ， 并 根据 基准 元 素 ( pivot element ) 将 这 个 栈 
分 为 两 个 栈 。 这 两 个 栈 会 进行 递归 排序 , 然后 归并 在 一 起 , 放 回 原来 的 栈 中 。 与 上 一 个 解法 一 样 ， 
每 层 递 归 都 会 创建 两 个 额外 的 栈 。 
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3.7 ”有 家 动物 收容 所 只 收容 狗 与 猫 ， 且 严格 遵守 “先进 先 出 ”的 原则 。 在 收养 该 收容 所 的 
动物 时 ， 收 养 人 只 能 收养 所 有 动物 中 “最 者 ”( 根据 进入 收容 所 的 时 间 长 短 ) 的 动物 ， 或 者 ， 可 
以 挑选 猫 或 狗 〈 同 时 必须 收养 此 类 动物 中 “最 者 ”的 )。 换 言 乙 ， 收 养 人 不 能 自由 挑选 想 收养 的 
对 象 。 请 创建 适用 于 这 个 系统 的 数据 结构 ， 实 现 各 种 操作 方法 ， 比 如 enqueue、dequeueAny、 
dequeueDog 和 和 dequeueCat 等 。 人 允许 使 用 Java 内 置 的 LinkedList 数 据 结 构 。( 第 51 页 ) 


解法 

这 个 问题 有 多 种 不 同 的 解法 。 比 如 ， 我 们 可 以 只 维护 一 个 队列 。 这 么 做 的 话 ，dequeueAny 
( 收养 任意 一 种 动物 ) 实现 起 来 很 简单 ,但 dequeueDog ( 收养 狗 ) 和 dequeueCat ( 收养 猫 ) 就 要 
迭代 访问 整个 队列 ， 才 能 找到 第 一 只 该 被 收养 的 狗 或 猫 。 这 会 增加 整个 解法 的 复杂 度 ， 降 低 执行 

另 一 种 解法 简单 明了 而 又 高 效 ， 只 需 为 狗 和 猫 各 自 创建 一 个 队列 ， 然 后 将 两 者 放 进 名 为 
AnimalQueue 的 包 庄 类 ， 并 且 存 储 某 种 形式 的 时 戳 ， 以 标记 每 只 动物 进入 队列 〈 即 收容 所 ) 的 时 
间 。 当 调用 dequeueAny 时 ， 查 看 狗 队列 和 猫 队列 的 首部 ， 并 返回 “最 老 ” 的 那 一 只 。 








1 public abstract class Animal { 

2 private int order; 

3 protected String name; 

4 public Animal(String n) { 

5 name = n; 

6 } 

4 

8 public void setOrder(int ord) { 

9 order = ord; 

16 } 

11 

12 public int getOrder() { 

13 return order; 

14 } 

15 

16 public boolean isOlderThan(Animal a) { 
17 return this.order < a.getOrder(); 
18 } 

19 } 

20 


21 public class AnimalQueue { 

22 LinkedList<Dog> dogs = new LinkedList<Dog>(); 
23 LinkedList<Cat> cats = new LinkedList<Cat>(); 
24 private int order = 86; // 用 作 时 发 


25 

26 public void enqueue(Animal a) { 

27 /* order 用 作 某 种 形式 的 时 稚 ， 以 便 比较 狗 或 猫 

28 * 插入 队列 的 先后 顺序 */ 

29 a.setOrder(order); 

36 order++; 

31 

32 if (a instanceof Dog) dogs.addLast((Dog) a); 

33 else if (a instanceof Cat) cats.addLast((Cat)a); 
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34 } 

35 

36 public Animal dequeueAny() { 
37 /* 查看 狗 和 猫 的 队列 的 首部 ， 弹 出 
38 * 最 旧 的 值 */ 

39 if (dogs.size() == 6) { 

46 return dequeueCats(); 

41 } else if (cats.size() == 6) { 
42 return dequeueDogs(); 

43 } 

44 

45 Dog dog = dogs.peek(); 

46 Cat cat = cats.peek(); 

47 if (dog.isOlderThan(cat)) { 
48 return dequeueDogs(); 

49 } else { 

56 return dequeueCats(); 

51 } 

S52 

53 public Dog dequeueDogs() { 

54 return dogs.poll(); 

55 } 

56 

57 public Cat dequeueCats() { 

58 return cats.poll(); 

59 } 

60 } 

61 


62 public class Dog extends Animal { 
63 public Dog(String n) { 

64 super(n); 

65 } 


68 public class Cat extends Animal { 
69 public Cat(String n) { 

76 super(n); 

71 } 


9.4 树 与 图 


4.1 实现 一 个 图 数 ， 检 查 二 叉 树 是 否 平衡 。 在 这 个 问题 中 ， 平 衡 树 的 定义 如 下 : 任意 一 个 
结 点 ， 其 两 棵 子 树 的 高 度 差 不 超过 1。( 第 54 页 ) 

解法 

还 算 幸 运 ， 此 题 至 少 明确 给 出 了 平衡 树 的 定义 : 任意 一 个 结 点 ， 其 两 棵 子 树 的 高 度 差 不 
大 于 1。 根 据 该 定义 可 以 得 到 一 种 解法 ， 即 直接 递归 访问 整 棵 树 ， 计 算 每 个 结 点 两 棵 子 树 的 


高 度 。 
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public static int getHeight(TreeNode root) { 
if (root == null) return 68; // 终止 条 件 
return Math.max(getHeight(root.1left), 
getHeight(root.right)) + 1; 
} 


1 
2 
3 
4 
5 
6 
7 public static boolean isBalanced(TreeNode root) { 

8 if (root == null) return true; // 终止 条 件 

9 

16 int heightDiff = getHeight(root.left) - getHeight(root.right); 


11 if (Math.abs(heightDiff) > 1) { 


12 return false; 

13 } else { // 递归 

14 return isBalanced(root.left) && isBalanced(root.right); 
15 } 

16 } 


虽然 可 行 , 但 效率 不 太 高 , 这 段 代码 会 递归 访问 每 个 结 点 的 整 棵 子 树 。 也 就 是 说 , getHeight 
会 被 反复 调用 计算 同一 个 结 点 的 高 度 。 因 此 ， 这 个 算法 的 时 间 复 杂 度 为 O(N log N)。 

我 们 可 以 删 减 部 分 getHeight 调 用 。 

仔细 查看 上 面 的 方法 ， 你 或 许 会 发 现 ，getHeight 不 仅 可 以 检查 高 度 ， 还 能 检查 这 棵 树 是 否 
平衡 。 那 么 ， 我 们 发 现 子 树 不 平衡 时 又 该 怎么 做 呢 ? 直接 返回 -1。 

改进 过 的 算法 会 从 根 结 点 递归 向 下 检查 每 棵 子 树 的 高 度 。 我 们 会 通过 checkHeight 方 法 ， 以 
递归 方式 获取 每 个 结 点 左右 子 树 的 高 度 。 若 子 树 是 平衡 的 ， 则 checkHeight 返 回 该 子 树 的 实际 高 
度 。 若 子 树 不 平衡 ， 则 checkHeight 返 回 -1。checkHeight 会 立即 中 断 执 行 ， 并 返回 -1。 

下 面 是 该 算法 的 实现 代码 。 
































public static int checkHeight(TreeNode root) { 
if (root == null) { 
return 6;j // 高 度 为 9 
} 


/* 检查 左 子 树 是 否 平 衡 */ 

int leftHeight = checkHeight(root.1left); 
if (leftHeight == -1) { 

9 return -1; // 不 平衡 

16 } 

1 /* 检查 左 子 树 是 否 平衡 */ 

12 int rightHeight = checkHeight(root.right); 


1 
这 
3 
4 
5 
6 
7 
8 


13 if (rightHeight == -1) { 
14 return -1; // 不 平衡 
15 } 

16 


17 /* 检查 当前 结 点 是 否 平衡 */ 
18 int heightDiff = leftHeight - rightHeight; 
19 if (Math.abs(heightDiff) > 1) { 





26 return -1; // 不 平衡 

21 } else { 

22 /* 返回 高 度 */ 

23 return Math.max(leftHeight, rightHeight) + 1; 
24 } 
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25 } 

26 

27 public static boolean isBalanced(TreeNode root) { 
28 if (checkHeight(root) == -1) { 

29 return false; 

36 } else { 

31 return true; 

32 } 

33 } 


这 段 代 码 需要 O(V) 的 时 间 和 CO(z) 的 空间 ， 其 中 厂 为 树 的 高 度 。 
4.2 ”给 定 有 向 图 ， 设 计 一 个 算法 ， 找 出 两 个 结 点 之 间 是 否 存在 一 条 路 径 。( 第 54 页 ) 


解法 

只 需 通 过 图 的 遍历 ， 比 如 深度 优先 搜索 或 广度 优先 搜索 等 ， 就 能 解决 这 个 问题 。 我们 从 两 个 
结 点 的 其 中 一 个 出 发 ,在 遍历 过 程 中 , 检查 是 否 找到 另 一 个 结 点 。 在 这 个 算法 中 , 访问 过 的 结 点 
都 应 标记 为 “已 访问 ”， 以 免 循环 和 重复 访问 结 点 。 

下 面 是 广度 优先 搜索 的 迭代 实现 。 


public enum State { 
Unvisited, Visited, Visiting; 


























} 


public static boolean search(Graph g, Node start, Node end) { 
// 当 作 队列 使 用 
LinkedList<Node> q = new LinkedList<Node>(); 


oONUUAWwWDOP 


for (Node u : g.getNodes()) { 
16 uU.state = State.Unvisited; 
} 

12 start.state = State.Visiting; 
13 q.add(start); 

14 Node u; 

15 while (!q.isEmpty()) { 


js 
pp 


16 U = q.removeFirst(); // 也 即 dequeue() 
17 if (u != null) { 

18 for (Node v : u.getAdjacent()) { 
19 if (v.state == State.Unvisited) { 
20 if (v == end) { 

21 return true; 

2 } else { 

23 Vv.state = State.Visiting; 
24 q.add(v); 

25 } 

26 } 

27 } 

28 u.state = State.Visited; 

29 } 

36 } 

3 return false; 

32 } 
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碰 到 这 类 问题 时 , 很 有 必要 跟 面 试 官 探讨 一 下 广度 优先 搜索 和 深度 优先 搜索 之 间 的 利 浆 。 例 
如 ,深度 优先 搜索 实现 起 来 比较 简单 ， 因 为 只 需 简单 的 递归 即 可 。 广 度 优先 搜索 很 适合 用 来 查找 
最 短路 径 ， 而 深度 优先 搜索 在 访问 邻近 结 点 之 前 ， 可 能 会 先 深 度 遍 历 其 中 一 个 邻近 结 点 。 


4.3 ”给 定 一 个 有 序 整 数 数 组 ， 元 素 各 不 相同 且 按 升序 排列 ， 编 写 一 个 算法 ,创建 一 棵 高 度 
最 小 的 二 又 查找 树 。( 第 54 页 ) 


解法 

要 创建 一 棵 高 度 最 小 的 树 ， 就 必须 让 左右 子 树 的 结 点 数量 越 接 近 越 好 。 也 就 是 说 ,我们 要 让 
数组 中 间 的 值 成 为 根 结 点 ， 这 么 一 来 ， 数 组 左边 一 半 就 成 为 左 子 树 ， 右 边 一 半 成 为 右 子 树 。 

然后 ,我 们 继续 以 类 似 方式 构造 整 棵 树 。 数 组 每 一 区 段 的 中 间 元 素 成 为 子 树 的 根 结 点 , 左 半 
部 分 成 为 左 子 树 ， 右 半 部 分 成 为 右 子 树 。 

一 种 实现 方式 是 使 用 简单 的 root .insertNode(int v) 方 法 ， 从 根 结 点 开始 ， 以 递归 方式 将 
值 v 插 入 树 中 。 这 么 做 的 确 能 构造 最 小 高 度 的 树 ， 但 效率 并 不 是 太 高 。 每 次 插入 操作 都 要 遍历 整 
棵 树 ， 时 间 开 销 为 O(N log NN)。 

另 一 种 做 法 是 以 递归 方式 运用 createMinimalBSsT 方 法 ， 从 而 消除 部 分 多 余 的 遍历 操作 。 这 
个 方法 会 传人 数组 的 一 个 区 段 ， 并 返回 最 小 树 的 根 结 点 。 

该 算法 简 述 如 下 。 

(1) 将 数组 中 间 位 置 的 元 素 插 入 树 中 。 

(2) 将 数组 左 半边 元 素 插入 左 子 树 。 

(3) 将 数组 右 半边 元 素 插入 右 子 树 。 

(4) 递归 处 理 。 

下 面 是 该 算法 的 实现 代码 。 




































































1 TreeNode createMinimalBST(int arr[], int start, int end) { 
2 if (end < start) { 

3 return null; 

4 } 

5 int mid = (start + end) / 2; 

6 TreeNode n = new TreeNode(arr[mid]); 

7 n.left = createMinimalBST(arr, start, mid - 1); 
8 n.right = createMinimalBST(arr, mid + 1, end); 
9 return n; 

16 } 

11 


12 TreeNode createMinimalBST(int array[]) { 
13 return createMinimalBST(array, 8, array.length - 1); 
14 } 


尽管 这 段 代码 看 起 来 不 是 特别 复杂 ， 但 在 编写 过 程 中 很 容易 犯 了 差 一 错误 (off-by-one )。 对 
这 部 分 代码 ， 务 必 进 行 详尽 测试 。 
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4.4 给 定 一 棵 二 叉 树 ， 设 计 一 个 算法 ， 创 建 含 有 某 一 深度 上 所 有 结 点 的 链表 〈 比 如， 若 一 
棵 树 的 深度 为 D， 则 会 创建 出 D 个 链表 )。( 第 54 页 ) 


解法 

乍 一 看 ,你 可 能 认为 这 个 问题 需要 一 层 一 层 逐 一 遍历 , 但 其 实 并 无 必要 。 你 可 以 用 任意 方式 
遍历 整 棵 树 ， 只 要 记 住 结 点 位 于 哪 一 层 即 可 。 

我 们 可 以 将 前 序 遍历 算法 稍 作 修改 ,将 level + 1 传人 下 一 个 递归 调用 。 下 面 是 使 用 深度 优 
先 搜索 的 实现 代码 。 
void createLevelLinkedList(TreeNode root, 


ArrayList<LinkedList<TreeNode>> lists, int level) { 
if (root == null) return; // 终止 条 件 





























户 


LinkedList<TreeNode> list = null; 
if (lists.size() == level) { // 该 层 不 在 链表 中 
list = new LinkedList<TreeNode>(); 
/* 以 中 序 遍 历 所 有 层级 ， 因 此 ， 若 这 是 第 一 次 
* 访问 第 i 层 ， 则 表示 我 们 已 访问 过 第 6 到 i-1 层 。 


ovam 上 wwN 


16 * 因此 ， 我 们 可 以 安全 地 将 这 一 层 加 到 链表 
11 * 末端 。 */ 

12 lists.add(list); 

13 } else { 

14 list = lists.get(level); 

15 } 


16 list.add(root); 
17 createLevelLinkedList(root.left, lists, level + 1); 
18 createLevelLinkedList(root.right, lists, level + 1); 


19 } 

20 

21 ArrayList<LinkedList<TreeNode>> createLevelLinkedList( 
22 TreeNode root) { 

23 ArrayList<LinkedList<TreeNode>> lists = 

24 new ArrayList<LinkedList<TreeNode>>(); 

25 createLevelLinkedList(root, lists, 0); 

26 return lists; 

27 } 


男 一 种 做 法 是 对 广度 优先 搜索 稍 加 修改 ， 即 从 根 结 点 开始 迭代 ， 然 后 第 2 层 ， 第 3 层 ， 等 等 。 

处 于 第 i 层 时 ， 则 表明 我 们 已 访问 过 第 订 1 层 的 所 有 结 点 。 也 就 是 说 ,要 得 到 i; 层 的 结 点 ， 只 需 
直接 查看 订 1 层 结 点 的 所 有 子 结 点 即 可 。 

下 面 是 该 算法 的 实现 代码 。 











1 ArrayList<LinkedList<TreeNode>> createLevelLinkedList( 
2 TreeNode root) { 
3 ArrayList<LinkedList<TreeNode>> result = 
4 new ArrayList<LinkedList<TreeNode>>(); 
5 /* 访问 根 结 点 */ 

6 LinkedList<TreeNode> current = new LinkedList<TreeNode>(); 
7 if (root != null) { 

8 current.add(root); 
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9 } 

16 

11 while (current.size() > 6) { 

12 result.add(current); // 加 入 上 一 层 

13 LinkedList<TreeNode> parents = current; // 转 到 下 一 层 
14 current = new LinkedList<TreeNode>(); 
15 for (TreeNode parent : parents) { 

16 /* 访问 子 结 点 */ 

17 if (parent.left != nul1) { 

18 current.add(parent.1eft); 

19 } 

26 if (parent.right != null) { 

21 current.add(parent.right); 

22 } 

23 } 

24 } 

25 return result; 

26 } 


你 可 能 会 问 , 这 两 种 解法 哪 一 种 效率 更 高 ?两 者 的 时 间 复 杂 度 皆 为 O(N), 那么 空间 效率 呢 ? 
乍 一 看 ， 我 们 可 能 会 以 为 第 二 种 解法 的 空间 效率 更 高 。 

在 某 种 意义 上 ， 这么 说 也 对 。 第 一 种 解法 会 用 到 O(log M) 次 递归 调用 (在 平衡 树 中 ), 每 次 调 
用 都 会 在 栈 里 增加 一 级 。 第 二 种 解法 采用 迭代 遍历 法 ， 不 需要 这 部 分 额外 空间 。 

不 过 ， 两 种 解法 都 要 返回 O(N) 数 据 ， 因 此 ， 北 归 实 现 所 需 的 额外 O(log 入 空间， 跟 必 须 传 回 
的 OOV) 数 据 相 比 , 并 不 算 多 。 虽然 第 一 种 解法 确实 使 用 了 较 多 的 空间 , 但 从 大 O 记 法 的 角度 来 看 ， 
两 者 效率 是 一 样 的 。 


4.5 “实现 一 个 函数 ， 检 查 一 棵 二 叉 树 是 否 为 二 叉 查找 树 。( 第 54 页 ) 


解法 
此 题 有 两 种 不 同 的 解法 。 第 一 种 是 利用 中 序 遍 历 , 第 二 种 则 建立 在 left <= current < right 
这 项 特性 之 上 。 


























解法 1: 中 序 遍 历 

看 到 此 题 , 闪 过 的 第 一 个 想法 可 能 是 中 序 遍历 , 将 所 有 元 素 复 制 到 数组 中 ,然后 检查 该 数组 
是 否 有 序 。 这 种 解法 要 多 用 一 点 内 存 ， 大 部 分 情况 下 都 没 问题 。 

唯一 的 问题 在 于 ， 它 无 法 正确 处 理 树 中 的 重复 值 。 例如 ,该 算法 无 法 区 分 下 面 这 两 棵 树 ( 其 
中 一 棵 是 无 效 的 )， 因 为 两 者 的 中 序 遍 历 结 果 相 同 。 


Valid BST [26.left = 26] 
Invalid BST [26.right = 26] 


不 过 , 要 是 假定 这 棵 树 不 得 包含 重复 值 ， 那么 这 种 做 法 还 是 行 之 有 效 的 。 该 方法 的 伪 码 大 致 
如 下 : 


1 public static int index = 0; 
2 public static void copyBST(TreeNode root, int[] array) { 
3 if (root == null) return; 








图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 








4 copyBST(root.left, array); 
5 array[index] = root.data; 

6 index++; 

7 copyBST(root.right, array); 
8 


} 
9 
16 public static boolean checkBST(TreeNode root) { 
11 int[] array = new int[root.size]; 


12 copyBST(root, array); 
13 for (int i = 1; i < array.length; i++) { 


14 if (array[i] <= array[i - 1]) return false; 
15 

16 return true; 

17 } 


注意 ， 这 里 必须 记录 数组 在 逻辑 上 的 “尾部 ”， 用 它 来 分 配 空间 以 储存 所 有 元 素 。 

仔细 审视 这 个 解法 , 我 们 就 会 发 现代 码 中 的 数组 实 无 必要 。 除 了 用 来 比较 某 个 元 素 和 前 一 个 
元 素 ， 别 无 他 用 。 那 么 ， 为 什么 不 在 进行 比较 时 ， 直 接 记 下 最 后 的 元 素 ? 

下 面 是 该 算法 的 实现 代码 。 


1 public static int last printed = Integer.MIN VALUE; 
2 public static boolean checkBST(TreeNode n) { 











3 if (n == null) return true; 

4 

5 // 递归 检查 左 子 树 

6 if (!checkBST(n.left)) return false; 
学 

8 // 检查 当前 结 点 

9 if (n.data <= last_printed) return false; 
16 last_printed = n.data; 

11 

12 // 递归 检查 右 子 树 

13 if (!checkBST(n.right)) return false; 
14 

15 return true; // 全 部 检查 完毕 

16 } 

















要 是 不 喜欢 使 用 静态 变量 ， 可 以 稍 作 修改 ， 使 用 包 囊 类 存放 这 个 整数 值 ， 如 下 所 示 : 


1 class WrapInt { 
2 public int value; 


3 } 
或 者 ， 若 用 C++ 或 其 他 支持 按 引 用 传 值 的 语言 实现 ， 就 可 以 这 么 做 。 


解法 2: 最 小 /最 大 法 

第 二 种 解法 利用 的 是 二 又 查找 树 的 定义 。 

一 棵 什么 样 的 树 才 成 其 为 二 叉 查 找 树 ”我 们 知道 这 棵 树 必 须 满足 以 下 条 件 : 对 于 每 个 结 点 ， 
left.data <= current.data < right.data, 但 是 这 样 还 不 够 。 试 看 下 面 这 棵 小 树 : 
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尽管 每 个 结 点 都 比 左 子 结 点 大 ， 比 右 子 结 点 小 , 但 这 显然 不 是 一 棵 二 又 查找 树 ， 其 中 25 的 位 
置 不 对 。 

更 准确 地 说 , 成 为 二 又 查找 树 的 条 件 是 : 所 有 左边 的 结 点 必须 小 于 或 等 于 当前 结 点 ， 而 当前 
结 点 必须 小 于 所 有 右边 的 结 点 。 

利用 这 一 点 , 我 们 可 以 通过 自 上 而 下 传递 最 小 和 最 大 值 来 解决 这 个 问题 。 在 迭代 遍历 整个 树 
的 过 程 中 ， 我 们 会 用 和 逐渐 变 罕 的 范围 来 检查 各 个 结 点 。 

以 下 面 这 棵 树 为 例 : 


个 5) 
中 7) 


首先 ， 从 (min = INT_MIN，max = INT_MAX) 这 个 范围 开始 ， 根 结 点 显然 落 在 其 中 。 然 后 处 
理 左 子 树 ， 检 查 这 些 结 点 是 否 落 在 (min = INT_MIN，max = 26) 范 围 内 。 然 后 再 处 理 ( 值 为 10 
的 结 点 ) 右 子 树 ， 检 查 结 点 是 否 落 在 (min = 26，max = INT_MAX) 范 于 内 。 

然后 ， 继 续 依 此 遍历 整 棵 树 。 进 入 左 子 树 时 ， 更 新 max。 进 入 右 子 树 时 ， 更 新 min。 只 要 有 任 
一 结 点 不 能 通过 检查 ， 则 停止 并 返回 false。 

这 种 解法 的 时 间 复 杂 度 为 OO), 其 中 入 为 整 棵 树 的 结 点 数 。 我 们 可 以 证 明 这 已 经 是 最 佳 做 法 ， 
因为 任何 算法 都 必须 访问 全 部 N 个 结 点 。 

因为 用 了 递归 , 对 于 平衡 树 , 空间 复杂 度 为 O(log M)。 在 调用 栈 上 , 共有 O(log 入 ) 个 递归 调用 ， 
因为 递归 的 深度 最 大 会 到 这 棵 树 的 深度 。 

该 解法 的 递归 实现 代码 如 下 : 
boolean checkBST(TreeNode n) { 


return checkBST(n, Integer.MIN VALUE, Integer.MAX VALUE); 


} 












































boolean checkBST(TreeNode n, int min, int max) { 
if (n == null) { 
return true; 


1 
2 
- 
4 
5 
6 
7 
8 } 
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9 if (n.data < min || n.data >= max) { 
16 Peturn false; 

11 } 

12 

13 if (!checkBST(n.left, min, n.data) || 
14 IcheckBST(n.right, n.data, max)) { 
15 return false; 

16 } 

17 return true; 

18 } 


记 住 ， 在 递归 算法 中 ,一 定 要 确定 终止 条 件 以 及 结 点 为 空 的 情况 得 到 受 善 处 理 。 


4.6 ”设计 一 个 算法 ， 找 出 二 又 查找 树 中 指定 结 点 的 “下 一 个 ” 结 点 ( 也 即 中 序 后 继 )。 可 
以 假定 每 个 结 点 都 含有 指向 父 结 点 的 连接 。( 第 54 页 ) 


解法 

回想 一 下 中 序 遍 历 ， 它 会 遍历 左 子 树 ， 然 后 是 当前 结 点 ， 接 着 是 右 子 树 。 要 解决 这 个 问题 ， 
必须 非常 小 心 ， 想 想 具 体 是 怎么 回 事 。 

假定 我 们 有 一 个 假想 的 结 点 。 我 们 知道 访问 顺序 为 左 子 树 ， 当 前 结 点 , 然后 是 右 子 树 。 显 然 ， 
下 一 个 结 点 应 该 位 于 右边 。 

不 过 , 到 底 是 右 子 树 的 哪个 结 点 呢 ? 如 果 中 序 遍 历 右 子 树 , 那 它 就 会 是 接 下 来 第 一 个 被 访问 
的 结 点 ， 也 就 是 说 ， 它 应 该 是 右 子 树 最 左边 的 结 点 。 够 简单 吧 ! 

但 是 ， 知 这 个 结 点 没有 右 子 树 ， 又 该 怎么 办 ? 这 种 情况 就 有 点 棘手 了 。 

若 结 点 n 没 有 右 子 树 ， 那 就 表示 已 遍 访 n 的 子 树 。 我 们 必须 回 到 n 的 父 结 点 ， 记 作 q。 

若 n 在 q 的 左边 ， 那 么 ， 下 一 个 我 们 应 该 访问 的 结 点 就 是 q( 中 序 遍 历 ，left -> current -> 
right )。 

若 n 在 q 的 右边 ， 则 表示 已 遍 访 q 的 子 树 。 我 们 需要 从 q 往 上 访问 ,直至 找到 我 们 还 未 完全 遍 访 
过 的 结 点 x。 怎么 才能 知道 还 未 完全 遍历 结 点 x 呢 ? 之 前 从 左 结 点 访问 至 其 父 结 点 时 , 就 已 碰 到 了 
这 种 情况 。 左 结 点 已 完全 遍历 ， 但 其 父 结 点 尚未 完全 遍历 。 
























































伪 码 如 下 : 

1 Node inorderSucc(Node n) { 

2 if (n has a right subtree) { 

3 return leftmost child of right subtree 
4 } else { 

5 while (n is a right child of n.parent) { 
6 n = n.parent; // 往 上 

7 ) 

8 return n.parent; // 父 结 点 还 未 遍历 

9 } 

16 } 


且慢 ， 如 果 一 路 往 上 裔 访 这 棵 树 都 没 发 现 左 结 点 呢 ?” 只 有 当 我 们 遇 到 中 序 遍 历 的 最 末端 时 ， 
会 出 现 这 种 情况 。 也 就 是 说 ， 如 果 我 们 已 位 于 树 的 最 右边 ， 那 就 不 会 再 有 中 序 后 继 ， 此 时 该 返 
回 nul1。 
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下 面 是 该 算法 的 实现 代码 (已 正确 处 理 结 点 为 空 的 情况 )。 
1 public TreeNode inorderSucc(TreeNode n) { 
2 if (n == null) return null; 

3 

4 /* 找到 右 子 结 点 ， 则 返回 右 子 树 里 

5 * 最 左边 的 结 点 */ 

6 if (n.right != null) 

7 return leftMostChild(n.right); 

8 } else { 

9 TreeNode q = ni 

16 TreeNode x = q.parent; 

11 // 向 上 直至 位 于 左边 而 不 是 右边 

12 while (x != null && x.left != q) { 
13 q = XxX; 

14 x = x.parent; 

15 } 

16 return x; 

17 } 

18 } 

19 


26 public TreeNode leftMostChild(TreeNode n) { 
21 if (n == null) { 


22 return null; 

23 } 

24 while (n.left != null) { 
25 n = n.left; 

26 } 

27 return n; 

28 } 

















这 不 是 世上 最 复杂 的 算法 问题 , 但 要 写 出 完美 无 瑕 的 代码 却 有 难度 。 面 对 这 类 问题 ， 比 较 实 
用 的 做 法 是 用 伪 码 勾勒 大 纲 ， 仔 细 描 绘 各 种 不 同 的 情况 。 


4.7 ”设计 并 实现 一 个 算法 ， 找 出 二 叉 树 中 某 两 个 结 点 的 第 一 个 共同 祖先 。 不 得 将 额外 的 结 
点 储存 在 另外 的 数据 结构 中 。 注 意 : 这 不 一 定 是 二 叉 查找 树 。( 第 54 页 ) 

解法 

如 果 是 二 又 查找 树 , 我 们 可 以 修改 find 操 作 , 用 来 查找 这 两 个 结 点 , 看 看 路 径 在 哪里 开始 分 
又。 可 惜 ， 这 不 是 二 又 查找 树 ， 因 此 必须 男 竟 他 法 。 

下 面 假定 我 们 要 找 出 结 点 p 和 q 的 共同 祖先 。 在 此 先 要 问 个 问题 , 这 棵 树 的 结 点 是 否 包含 指 问 
父 结 点 的 连接 。 


解法 1: 包含 指向 父 结 点 的 连接 

如 果 每 个 结 点 都 包含 指向 父 结 点 的 连接 ， 我 们 就 可 以 向 上 追踪 p 和 q 的 路 径 ， 直 至 两 者 相交 。 
不 过 ， 这 么 做 可 能 不 符合 题目 的 若干 假设 ， 因 为 它 需 要 满足 以 下 两 个 条 件 之 一 : (1) 可 将 结 点 标 
记 为 isvisited; (2) 可 用 另外 的 数据 结构 如 散 列 表 储存 一 些 数据 。 
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解法 2: 不 包含 指向 父 结 点 的 连接 

男 一 种 做 法 是 , 顺 着 一 条 p 和 q 都 在 同一 边 的 链子 , 也 就 是 说 , 若 p 和 q 都 在 某 结 点 的 左边 ,就 
到 左 子 树 中 查找 共同 祖先 。 若 都 在 右边 , 则 在 右 子 树 中 查找 共同 祖先。 要 是 p 和 q 不 在 同一 边 , 那 
就 表示 已 经 找到 第 一 个 共同 祖先 。 

这 种 做 法 的 实现 代码 如 下 。 


1 /* 若 p 为 root 的 子孙 ， 则 返回 true */ 
2 boolean covers(TreeNode root, TreeNode p) { 


























3 if (root == null) return false; 

4 if (root == p) return true; 

5 return covers(root.left, p) || covers(root.right, p); 
6 } 

到 

8 TreeNode commonAncestorHelper(TreeNode root，TreeNode p， 
9 TreeNode q) { 

16 if (root == null) return null; 

11 if (root == p || root == q) return root; 

12 


13 boolean is _p_on _ left 
14 boolean is _q_on_ left 


covers(root.1left, p); 
covers(root.1left, q); 


16 /* 若 p 和 q 不 在 同一 边 ， 则 返回 root */ 
17 if (is p_on left != is q_on left) return root; 


19 /* 否则 就 是 在 同一 边 ， 遍 访 那 一 边 */ 
26 TreeNode child side = is_p on left ? root.left : root.right; 


21 return commonAncestorHelper(child_side, p, q); 

22 } 

23 

24 TreeNode commonAncestor(TreeNode root, TreeNode p, TreeNode q) { 
25 if (!covers(root，p) || !covers(root，q)) { // 错误 检查 

26 return null; 

27 } 

28 return commonAncestorHelper(root, p, 9q); 

29 } 





这 个 算法 在 平衡 树 上 的 运行 时 间 为 0(n)。 这 是 因为 第 一 次 调用 时 ，covers 会 在 2n 个 结 点 上 调 
用 (左边 n 个 结 点 ,右边 n 个 结 点 )。 接 着 ,该 算法 会 访问 左 子 树 或 右 子 树 ， 此 时 covers 会 在 2n/2 
个 结 点 上 调用 ， 然 后 是 2n/4， 依 此 类 推 。 最 终 的 运行 时 间 为 O(n)。 

至 此 ， 就 渐 近 式 运 行 时 间 (asymptotic runtime ) 来 看 ， 可 以 确定 没有 更 优 解 了 ， 因 为 必须 遍 
访 这 棵 树 的 每 一 个 结 点 才 行 。 不 过 ， 或 许 我 们 还 能 减 小 常数 倍 的 值 。 

解法 3: 最 优化 

尽管 解法 2 在 运行 时 间 上 已 经 做 到 最 优 ， 还 是 可 以 看 出 部 分 低 效 的 操作 。 特 别 是 ，covers 会 
搜索 root 下 的 所 有 结 点 以 查找 p 和 q, 包括 每 棵 子 树 中 的 结 点 (root .left 和 root .right )。 然 后 ， 
它 会 选择 那些 子 树 中 的 一 棵 ， 搜 遍 它 的 所 有 结 点 。 每 棵 子 树 都 会 被 一 再 地 反复 搜索 。 

你 可 能 会 觉察 到 ， 只 需 搜索 一 遍 整 棵 树 ， 就 能 找到 p 和 q。 然 后 ， 就 可 以 “ 往 上 冒 泡 ”在 栈 里 
找到 先前 的 结 点 。 基 本 逻辑 与 上 一 种 解法 相同 。 
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使 用 函数 commonAncestor(TreeNode root，TreeNode p，TreeNode 9q) 递 归 访 问 整 棵 树 ， 
这 个 函数 的 返回 值 如 下 : 
口 返回 p， 若 root 的 子 树 含有 p( 而 非 q ); 
口 返回 q， 若 root 的 子 树 含 有 q( 而 非 p ); 
口 返回 nul1， 若 p 和 q 都 不 在 root 的 子 树 中 ; 
口 和 否则， 返回 p 和 q 的 共同 祖先 。 

在 最 后 一 种 情况 下 ， 要 找到 p 和 q 的 共同 祖先 比较 简单 。 当 commonAncestor(n.left，p，q) 
和 commonAncestor(n.right，p，9) 都 返回 非 空 的 值 时 ( 意 即 p 和 q 位 于 不 同 的 子 树 中 )， 则 n 即 
为 共同 祖先 。 

下 面 的 代码 提供 了 初步 的 解法 ， 不 过 其 中 有 个 bug。 试 着 找 找 看 。 


1 /* 下 面 的 代码 有 个 bug */ 
2 TreeNode commonAncestorBad(TreeNode root, TreeNode p, TreeNode q) { 
3 if (root == null) { 





4 return null; 

5 

6 if (root == p && root == q) { 

了 return root; 

8 } 

9 

16 TreeNode x = commonAncestorBad(root.1left, p, q); 

11 if (x != null && x != p && x != q) { // 已 经 找到 父系 结 点 
12 return x; 

13 } 

14 

15 TreeNode y = commonAncestorBad(root.right, p, q); 
16 if (y != null &&y != p &&y != q) { // 已 经 找到 父系 结 点 
17 return y; 

18 } 

19 

26 if (x != null && y != null) { // 在 不 同 子 树 里 找到 p 和 9q 
21 return root; // 这 是 共同 祖先 

22 } else if (root == p || root == q) { 

23 return root; 

24 } else { 

25 /* X 或 y 有 一 个 非 空 ， 则 返回 非 空 的 那个 值 */ 

26 return x == nul1 ? y : Xi; 

27 } 

28 } 


假如 有 个 结 点 不 在 这 棵 树 中 ， 这 段 代 码 就 会 出 问题 。 例 如 ， 请 看 下 面 这 棵 树 : 
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假设 我 们 调用 commonAncestor(node 3，node 5，node 7)。 当 然 ， 结 点 7 并 不 存在 ， 而 这 
正 是 问题 的 源头 。 调 用 序列 如 下 : 





1 commonAncestor(node 3, node 5, node 7) // --> 5 
2 calls commonAncestor(node 1, node 5, node 7) // --> null 
3 calls commonAncestor(node 5, node 5, node 7) // --> 5 
4 calls commonAncestor(node 8, node 5, node 7) // --> null 


换 句 话说 , 对 右 子 树 调 用 commonAncestor 时 , 前 面 的 代码 会 返回 结 点 5, 这 也 符合 代码 本 意 。 
可 题 在 于 查找 p 和 q 的 共同 祖先 时 ， 调 用 函数 无 法 区 分 下 面 两 种 4 人 
口 情况 1: p 是 q 的 子 结 点 ( 或 相反 ，q 为 p 的 子 结 点 )。 
口 情况 2: p 在 这 棵 树 中 ， 而 q 不 在 这 棵 树 中 (或 者 相反 )。 

不 论 哪 种 情况 ，commonAncestor 都 将 返回 p。 对 于 情况 1, 这 是 正确 的 返回 值 , 而 对 于 情况 2， 
返回 值 应 该 为 nul1。 

我 们 需要 设法 区 分 这 两 种 情况 ,这 也 是 以 下 代码 所 做 的 。 这 段 代 码 的 做 法 是 返回 两 个 值 : 结 
点 自身 ， 以 及 指示 这 个 结 点 是 否 确 为 共同 祖先 的 标记 。 


























} 


1 public static class Result { 

2 public TreeNode node; 

3 public boolean isAncestor; 

4 public Result(TreeNode n, boolean isAnc) { 
5 node = nj; 

6 isAncestor = isAnc; 

了 } 

8 

9 


16 Result commonAncestorHelper(TreeNode root, TreeNode p, TreeNode q){ 
11 if (root == null) { 


12 return new Result(null, false); 
13 } 

14 if (root == p && root == q) { 

15 return new Result(root, true); 

16 } 

17 

18 Result rx = commonAncestorHelper(root.left, p, q); 
19 if (rx.isAncestor) { // 找到 共同 祖先 
20 return rx; 

21 } 

22 


23 Result ry = commonAncestorHelper(root.right, p, q); 
24 if (ry.isAncestor) { // 找到 共同 祖先 


25 return ry; 

26 } 

27 

28 if (rx.node != null && ry.node != null) { 

29 return new Result(root,，true); // 这 是 共同 祖先 
36 } else if (root ==p || root == q) { 

31 /* 车 我 们 当前 位 于 p 或 9， 并 发 现 其 中 一 个 结 点 

32 * 位 于 子 树 中 ,那么 这 真 的 就 是 一 个 共同 祖先 ， 

33 # 标记 应 该 设 为 true。 */ 
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34 boolean isAncestor = rx.node != null || ry.node != null ? 

35 true : false; 

36 return new Result(root, isAncestor); 

37 } else { 

38 return new Result(rx.node!l=null ? rx.node : ry.node, false); 
39 } 

40 } 

41 

42 TreeNode commonAncestor(TreeNode root, TreeNode p, TreeNode q) { 
43 Result r = commonAncestorHelper(root, p, q); 

44 if (r.isAncestor) { 

45 return r.node; 

46 

47 return null; 

48 } 

当然 ， 由 于 这 个 问题 只 会 在 p 或 q 并 不 属于 这 棵 树 的 情况 下 出 现 ， 男 一 种 避免 bug 的 做 法 是 先 


搜 遍 整 棵 树 ， 以 确保 两 个 结 点 都 在 树 中。 


4.8 ”你 有 两 棵 非常 大 的 二 叉 树 : T1， 有 几 百 万 个 结 点 ; T2 ， 有 几 百 个 结 点 。 设 计 一 个 算 
法 ， 判 断 T2 是 否 为 T1 的 子 树 。 

如 果 T1 有 这 么 一 个 结 点 n， 其 子 树 与 T2 一 模 一 样 ， 则 T2 为 T1 的 子 树 。 也 就 是 说 ， 从 结 点 n 
处 把 树 砍 断 ， 得 到 的 树 与 T2 完全 相同 。( 第 54 页 ) 

解法 

磁 到 类 似 的 问题 ,不 妨 假设 只 有 少量 的 数据 ， 以 此 为 基础 解决 问题 。 这么 做 很 有 用 ,可 以 借 
此 找 出 可 行 的 基本 解法 。 

在 规模 较 小 且 较 简单 的 问题 中 ， 我 们 可 以 创建 一 个 字符 串 ， 表 示 中 序 和 前 序 遍 历 。 若 T2 
前 序 遍 历 是 TI1 前 序 遍 历 的 子囊 ， 并 且 T2 中 序 遍 历 是 T1 中 序 遍 历 的 子囊 ， 则 T2 为 T1 的 子 树 。 利 
用 后 绥 树 可 以 在 线性 时 间 内 检查 是 否 为 子囊 ， 因 此 就 最 差 情 况 的 时 间 复 杂 度 而 言 ， 这 个 算法 是 
相当 高 效 的 。 

注意 , 我 们 需要 在 字符 串 中 搬入 特殊 字符 ， 表 示 左 结 点 或 右 结 点 为 NULL 的 情况 。 和 否则, 我 们 
就 无 法 区 分 以 下 两 种 情况 : 












































T1 T2 

尽管 这 两 棵 树 不 同 ,但 两 者 的 中 序 和 前 序 遍 历 完全 一 样 。 
T1， 中 序 : 3，3 
T1， 前 序 : 3，3 
T2， 中 序 : 3，3 


2， 前 序 : 3， 3 


不 过 ， 要 是 标记 出 NULL 值 ， 就 能 区 分 这 两 棵 树 : 
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T1， 中 序 : 6, 3, 0, 3, 0 
T1， 前 序 : 3,，3, 6, 6, 0 
T2， 中 序 : 6, 3, 0, 3, 0 
T2， 前 序 : 3, 60,3, 86, 0 

















对 于 简单 的 情形 ， 这 种 解法 还 算 不 错 ， 但 是 我 们 真正 要 面 对 的 问题 涉及 的 数据 量 要 大 得 多 。 
鉴于 该 问题 指定 的 约束 条 件 ， 创 建 两 棵 树 的 副本 可 能 要 占用 太 多 的 内 存 。 


另 一 种 解法 

另 一 种 解法 是 搜 遍 较 大 的 那 棵 树 T1 。 每 当 Tl 的 某 个 结 点 与 T2 的 根 结 点 匹配 时 ， 就 调用 
treeMatch。treeMatch 方 法 会 比较 两 棵 子 树 ， 检 查 两 者 是 否 相 同 。 

分 析 运 行 时 间 有 点 复杂 , 粗略 一 看 的 答案 可 能 是 O(nm)， 其 中 n 为 T1 的 结 点 数 ,，m 为 T2 的 结 点 
数 。 虽 然 在 技术 上 这 个 答案 是 正确 的 ， 但 稍微 再 想 想 就 能 得 到 更 靠 谱 的 答案 。 

我 们 不 必 对 T2 的 每 个 结 点 调用 treeMatch, 而 是 会 调用 次 ， 其 中 为 T2 根 结 点 在 T1 中 出 现 的 
次 数 。 因 此 运行 时 间接 近 O(n + km)。 

其 实 ， 即 使 这 样 运行 时 间 也 有 所 等 大。 即使 根 结 点 相同 ,一旦 发 现 T1 和 T2 有 结 点 不 同 ， 我 
们 就 会 退出 treeMatch。 因 此 ， 每 次 调用 treeMatch， 也 不 见得 都 会 查看 m 个 结 点 。 

下 面 是 该 算法 的 实现 代码 。 















































1 boolean containsTree(TreeNode t1, TreeNode t2) { 
2 if (t2 == null) { // 空 树 一 定 是 子 树 

3 return true; 

4 } 

5 return subTree(t1, t2); 

6 } 

7 

8 boolean subTree(TreeNode r1, TreeNode r2) { 

9 if (rl == null) { 

16 return false; // 大 的 树 已 经 室 了 ， 还 未 找到 子 树 
11 } 

12 if (rl.data == r2.data) { 

13 if (matchTree(r1,r2)) return true; 

14 

15 return (subTree(r1.left, r2) || subTree(r1.right, r2)); 
16 } 

17 


18 boolean matchTree(TreeNode r1l, TreeNode r2) { 
19 if (r2 == null] && r1 == null) // 车 两 者 都 空 
26 return true; // 子 树 中 已 无 结 点 


22 // 若 其 中 之 一 为 空 ， 但 并 不 同时 为 空 
23 if (rl == null || r2 == null) { 


24 return false; 

25 } 

26 

27 if (rl.data != r2.data) 

28 return false; // 结 点 数据 不 匹配 

29 return (matchTree(r1.left, r2.left) && 
36 matchTree(r1.right, r2.right)); 
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31 } 
32 } 


什么 情况 下 用 简单 解法 比较 好 , 什么 时 候 另 一 种 解法 比较 好 呢 ? 这 个 问题 值得 跟 面 试 官 好 好 
讨论 一 番 ， 下 面 是 几 点 注意 事项 。 

(1) 简单 解法 会 占用 OU + 由 内 存 ， 而 另 一 种 解法 则 占用 OUdog(o + log(m)) 内 存 。 记 住 : 要 求 
可 扩展 性 时 ， 内 存 使 用 多 朝 关系 重大 。 

(2) 简单 解法 的 时 间 复 杂 度 为 O(n + m)， 男 一 种 解法 在 最 差 情 况 下 的 执行 时 间 为 Om)。 话 说 
回来 ， 只 看 最 差 情 况 的 时 间 复 杂 度 可 能 会 造成 误导 ， 我 们 需要 做 进一步 观察 。 

(3) 如 前 所 述 ， 比 较 准 的 运行 时 间 为 O(n + km)， 其 中 为 T2 根 结 点 在 T1 中 出 现 的 次 数 。 假 设 
Tl1 和 T2 的 结 点 数据 为 0 和 p 之 间 的 随机 数 ， 则 K 值 大 约 为 n/p， 为 什么 ”因为 TT 有 nn 个 结 点 ,每 个 结 
点 有 1 的 几率 与 T2 根 结 点 相同 ， 因 此 ，T1 中 大 约 有 7 V/ p 个 结 点 等 于 T2 根 结 点 ( T2.root )。 举 个 
例子 ， 假 设 p = 1666, n = 1 666 68666 日 m = 166。 我 们 需要 检查 的 结 点 数量 大 致 为 1 100 000 (1 
168 668 = 1 68 8868 + 166*1 68 868/1666 )。 

(4) 借助 更 复杂 的 数学 运算 和 假设 ， 就 能 得 到 更 准确 的 运行 时 间 。 在 第 3 点 中 ， 我 们 假设 调 
用 treeMatch 时 将 遍历 T2 的 全 部 m 个 结 点 。 然 而 ， 更 有 可 能 出 现 的 情况 是 ， 我 们 很 早 就 发 现 两 棵 
树 有 不 同 的 结 点 ， 然 后 提早 就 退出 了 这 个 函数 。 

总 的 来 说 ,在 空间 使 用 上 ， 另 一 种 解法 显然 比较 好 , 在 时 间 复 杂 度 上 ,也 可 能 比 简单 解法 更 
优 。 一 切 都 取决 于 你 做 出 哪些 假设 ， 以 及 要 不 要 考虑 牺牲 最 差 情 况 的 运行 时 间 , 来 减少 平均 情况 
的 运行 时 间 。 这 一 点 非常 值得 向 面试 官 提出 并 讨论 。 



























































4.9 给 定 一 棵 二 叉 树 ， 其 中 每 个 结 点 都 含有 一 个 数值 。 设 计 一 个 算法 ， 打 印 结 点 数值 总 和 
等 于 某 个 给 定 值 的 所 有 路 径 。 注 意 , 路 径 不 一 定 非 得 从 二 叉 树 的 根 结 点 或 叶 结 点 开始 或 结束 。( 第 
54 页) 

解法 

下 面 我 们 运用 简化 推广 法 来 解 题 。 

部 分 1: 简化 一 一 假设 路 径 必须 从 根 结 点 开始 ， 但 可 以 在 任意 结 点 结束 ， 怎 么 解决 ? 








在 这 种 情况 下 ， 问 题 就 会 变 得 容易 很 多 。 

我 们 可 以 从 根 结 点 开始 ,向 左 向 右 访 问 子 结 点 ,计算 每 条 路 径 上 到 当前 结 点 为 止 的 数值 总 和 ， 
若 与 给 定 值 相同 则 打印 当前 路 径 。 注 意 ， 就 算 找到 总 和 ， 仍 要 继续 访问 这 条 路 径 。 为 什么 ?因为 
这 条 路 径 可 能 继续 往 下 经 过 a + 1 结 点 和 a - 1 结 点 (或 其 他 数值 总 和 为 0 的 结 点 序列 )， 完 整 路 
径 的 总 和 仍然 等 于 sum。 

例如 ， 和 若 sum = 5， 可 能 会 得 到 以 下 路 径 : 
Dp = {2,，3} 
DDq= {2, 3, -4, -2, 6} 

如 果 找 到 2 + 3 就 停 下 来 ,我们 就 会 错过 第 二 条 路 径 ， 还 可 能 错过 其 他 路 径 。 因 此 ， 我 们 必 
须 继续 往 下 查找 所 有 可 能 的 路 径 。 
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部 分 2: 推广 一 一 路 径 可 从 任意 结 点 开始 。 

现在 ,如 果 路 径 可 从 任意 结 点 开始 , 该 怎么 办 ? 在 这 种 情况 下 ,我 们 可 以 稍 作 调整 。 对 于 每 
个 结 点 ,我们 都 会 向 “上 ”检查 是 否 得 到 相符 的 总 和 。 也 就 是 说 ,我 们 不 再 要 求 “ 从 这 个 结 点 开 
始 是 否 会 有 总 和 为 给 定 值 的 路 径 ”, 而 是 关注 “这 个 结 点 是 否 为 总 和 为 给 定 值 的 某 条 路 径 的 末端 ”。 

递归 访问 每 个 结 点 n 时 , 我 们 会 将 root 到 n 的 完整 路 径 传 人 该 函数 。 随 后 , 这 个 函数 会 以 相反 
的 顺序 ， 从 n 到 root ， 将 路 径 上 的 结 点 值 加 起 来 。 当 每 条 子路 径 的 总 和 等 于 sum 时 ， 就 打印 这 条 
路 径 。 
































1 public void findsum(TreeNode node, int sum, int[] path, int level) { 
2 if (node == null) { 

3 return; 

4 } 

5 

6 /* 将 当前 结 点 插入 路 径 */ 

了 path[level] = node.data; 

8 

9 /* 查找 以 此 为 终点 且 总 和 为 Sum 的 路 径 */ 

16 int t = 9; 

11 for (int i = level; i >= 6; i--){ 

12 t += path[i]; 

13 if (t == sum) { 

14 print(path, i, level); 

15 } 

16 } 

17 

18 /* 查找 此 结 点 之 下 的 结 点 */ 

19 findSum(node.left, sum, path, level + 1); 
20 findSum(node.right, sum, path, level + 1); 
了 于 

22 /* 从 路 径 中 移 除 当 前 结 点 。 严 格 来 说 并 不 一 定 要 这 么 做 ， 
23 * 直接 忽略 这 个 值 即 可 ， 但 这 么 做 是 个 好 习惯 */ 
24 path[level] = Integer.MIN_VALUE; 

25 } 

26 


27 public void findSum(TreeNode node, int sum) { 
int depth = depth(node); 
29 int[] path = new int[depth]; 


DD 
Oo 


36 findsum(node，sum，path，6); 

31 } 

32 

33 public static void print(int[] path, int start, int end) { 
34 for (int i = start; i <= end; i++) { 

35 System.out.print(path[i] + “ “); 

36 } 

37 System.out.println(); 

38 } 

39 


46 public int depth(TreeNode node) { 
41 if (node == null) { 

42 return 0@; 

43 } else { 
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44 return 1 + Math.max(depth(node.left), depth(node.right)); 
45 } 
46 } 


那么 ， 这 个 算法 的 时 间 复 杂 度 如 何 (假设 是 棵 平衡 二 义 树 ) ?如 果 结 点 在 r 层 ， 那 么 就 需要 x 
份量 的 工作 (向 “上 ”检查 结 点 的 步 又 )。 我 们 可 以 猜测 时 间 复 杂 度 为 O(n log(n))， 因 为 总 共有 nn 
个 结 点 ,平均 下 来 ， 每 一 步 需 要 log(n) 的 工作 量 。 

如 果 这 么 分 析 ， 你 还 是 看 不 大 明白 ， 我 们 也 可 以 用 严格 的 数学 推导 来 说 明 。 注 意 ， 在 r 层 上 
有 2' 个 结 点 。 

1 *21+2*22+3*23 4+4*24+...d*27 
= sum(r * 2", r from 0 to depth) 
=2*(d-1)*2 +2 

n=24 

d= log(n) 

注意 ，2” 中 =x， 因 此 ， 

O(2 * (log(n) - 1) * 2 + 2) 
= O(2 (logn-1)*n) 
= O(n log(n)) 

按照 同样 的 逻辑 ， 可 以 推导 出 算法 的 空间 复杂 度 为 O(log(n))， 因 为 该 算法 会 递归 O(logn) 次 ， 
而 在 递归 调用 中 参数 path 只 分 配 一 次 空间 (大 小 为 O(log n) )。 


9.5 ”位 操作 


5.1 ”给 定 两 个 32 位 的 整数 N 与 M， 以 及 表示 比特 位 置 的 /与 j/。 编 写 一 个 方法 ， 将 M 插 入 N， 
使 得 M 从 的 第 /位 开始 ， 到 第 /位 结束 。 假 定 从 位 到 位 足以 容纳 M， 也 即 若 M=10011， 那 么 /和 这 间 
至 少 可 容纳 5 个 位 。 例 如 ， 不 可 能 出 现 / = 3 和 / = 2 的 情况 ， 因 为 第 3 位 和 第 2 位 之 间 放 不 下 M。 

示例 输入 : N = 10000000000, M = 10011, i= 2, /= 6 输出 : N= 10001001100 (第 56 页 ) 


解法 

这 个 问题 的 解决 可 分 为 三 大 步骤 。 

(1) 将 N 中 从 /到 这 间 的 位 清 零 。 

(2) 对 MM 执行 移 位 操作 ， 与 j 和 i 之 间 的 位 对 齐 。 

(3) 合并 M 与 N。 

其 中 步骤 1 最 为 坏 手 。 如 何 将 N 中 的 那些 位 清 零 呢 ? 我 们 可 以 利用 掩 码 来 清 零 。 除 /到 i 这 间 的 
位 为 0 外 ， 这 个 掩 码 的 其 余 位 均 为 1。 我 们 会 先 创建 掩 码 的 左 半 部 分 ， 然 后 是 右 半 部 分 ， 最 终 得 到 
整个 掩 码 。 

1 int updateBits(int n, int m, int i, int j) { 


2 /* 创建 挫 码 ， 用 来 清除 n 中 到 j 的 位 
/* 示例 : i = 2，j = 4。 掩 码 为 11168811。 
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4 * 为 简单 起 见 ， 本 例 掩 码 只 有 8 位 

5: */ 

6 int allOnes = ~6j // 等 同 于 一 连 串 的 1 

7 

8 // 在 位 置 j 之 前 的 位 均 为 1， 其 余 为 96，left = 11166666 
9 int left = allOnes << (j + 1); 

16 


11 // 在 位 置 i 之 后 的 位 均 为 1, right = 68888811 
12 int right = ((1 << i) - 1); 


14 // 除 并 到 j 的 位 为 G8， 其 余 位 均 为 1。mask = 11166611 
15 int mask = left | right; 


17 /* 清除 位 置 j 到 i 的 位 ， 然 后 将 m 放 进去 */ 
18 int n_cleared = n & mask; // 清除 j 到 i 的 位 
19 int m_shifted = m << i; // 将 m 移 至 相应 的 位 置 


20 
21 return n_cleared | m_shifted; // 对 两 者 执行 位 或 操作 ， 搞 定 ! 
22 } 





解决 这 类 问题 时 ( 包括 许多 位 操作 问题 )， 务 必 切 实 充 分 地 对 代码 进行 测试 。 和 否则 ， 一 不 小 


心 就 容易 犯 下 差 一 错误 。 


5.2 给 定 一 个 介 于 0 和 1 之 间 的 实数 (如 0.72 )， 类 型 为 double， 打 印 它 的 二 进 制 表示 。 如 
果 该 数字 无 法 精确 地 用 32 位 以 内 的 二 进 制 表示 ， 则 打印 “ERROR”*。( 第 56 页 ) 


解法 





注意 ， 为 表示 清晰 起 见 ， 这 里 分 别 用 x, 和 xio 来 指示 x 是 二 进 制 还 是 十 进 制 。 




















首先 ， 我们 要 和 弄 清楚 非 整 型 的 数字 用 二 进 制 表示 是 什么 样 的 。 与 十 进 制 数 相仿 ， 二 进 制 数 





0.101> 表 示 如 下 : 
0.101;= 1 * (1/2')+0*(1/2") + 1*(1/2°) 





为 了 打印 小 数 部 分 , 我 们 可 以 将 这 个 数 乘 以 2, 检查 2n 是 否 大 于 或 等 于 1。 这 


动 ” 小 数 部 分 ， 也 即 : 
r=210*n 
=210* 0.1012 
=1*(1/2") +0*(1/2')+1*(1/27) 
=1.01， 


实质 上 等 同 于 “ 移 


若 r >= 1， 可 知 n 的 小 数 点 后 面 正好 有 个 1。 不 断 重复 上 述 步骤 ， 我 们 可 以 检查 每 个 数位 。 


public static String printBinary(double num) { 
if (num >= 1 || num <= 6) { 
return “ERROR”; 


} 


StringBuilder binary = new StringBuilder(); 
binary.append(".”); 
while (num > 9) { 


本 
2 
3 
4 
5 
6 
pi 
8 
9 /* 设 定 长 度 上 限 : 32 个 字符 */ 
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16 if (binary.length() >= 32) { 
11 return “ERROR”; 

12 } 

13 

14 double r = num * 2; 

15 if (r >= 1) { 

16 binary.append(1); 

17 num = r -1; 

18 } else { 

19 binary.append(0); 

26 num = r; 

21 } 

22 } 

23 return binary.toString(); 
24 } 


上 面 的 做 法 是 将 数字 乘 以 2， 然 后 与 1 进行 比较 ， 此 外 我 们 还 可 以 将 这 个 数 与 0.5 比 较 ， 然 后 
与 0.25 比 较 ， 依 此 类 推 。 下 面 的 代码 示范 了 这 一 做 法 。 


1 public static String printBinary2(double num) { 
2 if (num >= 1 || num <= 6) { 

3 return “ERROR”; 

4 } 

5 

6 StringBuilder binary = new StringBuilder(); 
到 double frac = 8.5; 

8 binary.append(“.2”); 

9 while (num > 6) { 

16 /* 设 定 长 度 上 限 : 32 个 字符 */ 
11 if (binary.length() > 32) { 
12 return “ERROR”; 

13 

14 if (num >= frac) { 

15 binary.append(1); 

16 num -= frac; 

17 } else { 

18 binary.append(0); 

19 } 

26 frac /= 2; 

21 } 

22 return binary.toString(); 

23 } 


这 两 种 做 法 都 很 不 错 ; 具体 怎么 做 ， 就 看 你 个 人 觉得 哪 种 做 法 更 自然 。 

不 论 采 用 哪 种 方式 , 对 于 这 类 问题 , 一 定 要 准备 好 详尽 的 测试 用 例 , 并 在 面试 中 切实 进行 测试 。 

5.3 ”给 定 一 个 正 整数 ， 找 出 与 其 二 进 制 表示 中 1 的 个 数 相 同 、 且 大 小 最 接近 的 那 两 个 数 
(一 个 略 大 ， 一 个 略 小 )。( 第 56 页 ) 

解法 

这 个 问题 有 多 种 解法 ,包括 蛮 力 法 、 位 操作 以 及 巧妙 运用 算术 。 注 意 , 运用 算术 法 建立 在 位 
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操作 的 解法 之 上 。 在 介绍 算术 方法 之 前 ， 你 应 该 先 学 会 位 操作 的 解法 。 


1. 蛮 力 法 

简单 的 做 法 就 是 直接 使 用 蛮 力 : 在 n 的 二 进 制 表示 中 ， 数 出 1 的 个 数 ， 然 后 增加 或 减 小 ， 直 至 
找到 1 的 个 数 相 同 的 数字 。 简 单 吧 ， 但 也 没什么 意思 。 还 有 没有 更 优 的 做 法 呢 ? 当然 有 ! 

下 面 先 从 getNext 的 代码 开始 ， 然 后 是 getPrev。 


2. 位 操作 法 : 取得 后 一 个 较 大 的 数 
要 是 你 还 在 考虑 后 一 个 数 应 该 是 什么 样 的 ， 不 妨 作 如 下 观察 。 以 数字 13 948 为 例 ， 二 进 制 表 
示 如 下 : 













































































我 们 想 让 这 个 数 大 一 点 (但 又 不 会 太 大 )， 同 时 1 的 个 数 又 要 保持 想 不 变 。 

你 会 发 现 : 给 定 一 个 数 x 和 两 个 位 的 位 置 和 上 j/， 假 设 将 位 i 从 1 翻转 为 0， 位 } 从 0 翻转 成 1。 若 i> 
J，n 就 会 减 小 ; 若 i<j， 则 n 就 会 变 大 。 

继而 得 到 以 下 几 点 。 

(1) 若 将 某 个 0 翻转 成 1， 就 必须 将 某 个 1 翻转 为 0。 

(2) 进行 位 翻转 时 ， 如 果 0 变 1 的 位 处 于 1 变 0 的 位 的 左边 ， 这 个 数字 就 会 变 大 。 

(3) 我 们 想 让 这 个 数 变 大 , 但 又 不 致 大 大 。 因 此 , 必须 翻转 最 右边 的 0, 且 它 的 右边 必须 还 有 个 1。 

换 句 话 说 , 我 们 要 翻转 最 右边 但 非 拖 尾 的 0。 用 上 面 的 例子 来 说 , 拖 尾 0 位 于 第 0 到 第 1 个 位 置 。 
因此 ， 最 右边 但 不 是 拖 尾 的 0 处 在 位 置 7。 我 们 把 这 个 位 置 记 作 p。 

@ 步骤 1: 翻转 最 右边 、 非 拖 尾 的 0 




















1 
13 


1 
12 


1 
5 


9 
1 


9 
9 


o|1|1 
11|10|9 


将 位 置 7 翻转 后 ，n 就 会 变 大 。 但 是 ， 现 在 n 中 的 1 多 了 一 个 ，0 少 了 一 个 。 我 们 还 需 尽 量 缩小 
数值 ， 同 时 记得 满足 要 求 。 

缩小 数值 时 ， 可 以 重新 排列 位 p 右 方 的 那些 位 ， 其 中 ，0 放 到 左边 ，1 放 到 右边 。 在 重新 排列 
的 过 程 中 ， 还 要 将 其 中 一 个 1 改 为 0。 

有 种 相对 简单 的 做 法 是 , 数 出 p 右 方 有 几 个 1, 将 位 置 0 到 位 置 p 的 所 有 位 清 零 , 然后 回填 c1-1 
个 1。 假设 c1 为 p 右 方 1 的 个 数 ，ce 为 p 右 方 0 的 个 数 。 

下 面 举 例 说 明 这 些 操 作 。 

@ 步骤 2: 将 p 右 方 的 所 有 位 清 堆 ， 由 步骤 1 可 知 ，c9 = 2，c1 = 5, p=7 








1|1|1 
EE 

















1 1 
7 6 
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为 了 将 这 些 位 清 零 ， 和 需要 创建 一 个 掩 码 ， 前 面 是 一 连 串 的 1， 后 面 跟 着 p 个 0， 做 法 如 下 : 


a=1<x<p; // 除 位 p 为 1 外 ， 其 余 位 均 为 6 
b=a-1; // 前 面 侈 为 9， 后 面 跟 p 个 1 
mask = ~b; // 前 面 全 为 1， 后面 跟 p 个 @ 
n = n & mask; // 将 右边 p 个 位 清 零 


或 者 ， 更 简洁 的 做 法 是 : 
n &= ~((1 << p) - 1); 
@ 步骤 3: 回填 cl1 - 1 个 1 











1|1|16|1|1|6|i|1|16|96|191|11|1|11 11 
13 |1211411|161|19 za 3 | ne 
要 在 p 右 边 插入 cl - 1 个 1， 做 法 如 下 : 
a=1<x< (cl -1); // 位 cl - 1 为 1， 其 余 位 均 为 6 
b=a-1; // 位 6 到 位 c1 - 1 的 位 为 1， 其 余 位 均 为 8 
n=n| bi // 在 位 6 到 位 c1 - 1 处 插入 1 


或 者 ， 更 简洁 一 点 : 

n |= (1 «<< (cl1 - 1)) - 1; 

至 此 ， 我 们 得 到 大 于 n 的 数字 中 ，1 的 个 数 与 n 的 相同 的 最 小 数字 。 
getNext 的 实现 代码 如 下 : 





1 public int getNext(int n) { 

2 /* 计算 c@ 和 c1 */ 

3 int c = nj; 

4 int c@ = 0; 

5 int cl = 0@; 

6 while (((c & 1) == 6) && (c != 6)) { 

己 CQ++; 

8 Cc >>= 1; 

9 } 

16 

11 while ((c & 1) == 1) { 

12 C1l++; 

13 C >>= 1; 

14 } 

15 

16 /* 错误 : 若 n == 11..1160...68， 那么 就 没有 更 大 的 数字 ， 
17 * 且 1 的 个 数 相 同 */ 

18 if (ce+cl==31 || ce+cl== 6) 1{ 

19 return -1; 

26 } 

21 

22 int p = c@ + c1; // 最 右边 、 非 抑 尾 8 的 位 置 

23 

24 nl|= (1 << p); // 翻转 最 右边 、 非 拖 尾 9 

25 nn &= ~((1 << p) - 1); // 将 p 右 方 的 所 有 位 清 零 
26 nn |= (1 << (cl - 1)) - 1; // 在 右 方 插入 (c1-1) 个 1 
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27 return n; 
28 } 


3. 位 操作 法 : 获取 前 一 个 较 小 的 数 

getPrev 的 实现 方法 与 getNext 的 非常 相似 。 

(1) 计算 ce 和 c1。 注 意 c1 是 拖 尾 1 的 个 数 ， 而 ce 为 紧邻 拖 尾 1 的 左 方 一 连 串 0 的 个 数 。 
(2) 将 最 右边 、 非 拖 尾 1 变 为 0， 其 位 置 为 p = c1 + ce。 

(3) 将 位 p 右 边 的 所 有 位 清 零 。 

(4) 在 紧邻 位 置 p 的 右 方 ， 插 入 cl + 1 个 1。 

注意 ， 步 又 2 将 位 p 清 零 ， 而 步骤 3 将 位 0 到 位 p-1 清 零 ， 我 们 可 以 将 这 两 步 合并 。 
下 面 举 例 说 明 各 个 步骤 。 


















































@ 步骤 1: 初始 数字 , p = 7，c1 = 2, c@ = 5 
1 9 9 1 1 1 1 0 9 9 9 9 1 1 
1) | 2 | | ed | a ea sa | rp | es | | | 
@ 步骤 2、3: 将 位 0 到 位 p 清 堆 
1 9 9 1 1 1 9 0 9 9 9 9 0 9 
Ta | | | | ea | ,ea | ss | rp | Es | | a | 
具体 做 法 如 下 所 示 : 
int a = ~@; // 所 有 位 置 1 
int b = a xc (p + 1); // 位 p 左 方 的 所 有 位 为 1 后跟 p+1 个 0 
n &= b; // 将 位 8 到 位 p 清 零 
@ 步骤 4: 在 紧邻 位 置 p 的 右 方 ， 插 入 c1 + 1 个 1 
1 9 0 1 1 1 9 1 1 1 9 9 0 0 
.|| 2 a | |, Ne | ss | ip | Es | 














注意 ，p = cl + c6， 因 此 (cl + 1) 个 1 的 后 面 会 跟 (c@ - 1) 个 0。 


位 (cl + 1) 为 1， 其 余 位 均 为 6 
前 面 为 9， 后 面 跟 cl + 1 个 1 
Cc1+1 个 1， 后 面 跟 c8-1 个 6 


1<< (cl + 1); // 
a-1; Xi 
b << (ce - 1); // 


getPrev 的 实现 代码 如 下 所 示 。 


1 
2 


int getPprev(int n) { 
int temp = n; 
int c@ = 0 
int c1 = 0 
while (temp & 1 
C1++; 
temp >>= 1; 


3 
4 
5 1) { 
6 
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8 } 
16 if (temp == 6) return -1; 


12 while (((temp & 1) == 06) && (temp != 6)) { 
13 CO++; 

14 temp >>= 1; 

15 } 


17 int p = c@ + cl; // 最 右边 、 非 拖 尾 1 的 位 置 
18 n &= ((~6) < (p + 1)); // 将 位 9 到 位 p 清 堆 


26 int mask = (1 << (cl + 1)) - 1; // (cil+1) 个 1 
21 n |= mask << (ce - 1); 


23 return n; 


4. 算术 解法 获取 后 一 个 数 

如 果 ce 是 拖 尾 0 的 个 数 ，c1 是 拖 尾 0 左 方 全 为 1 的 位 的 个 数 ， 而 且 p = ce + c1， 于 是 我 们 就 
可 以 将 前 面 的 解法 表述 如 下 。 

(1) 将 位 p 置 1。 

(2) 将 位 0 到 位 p 清 零 。 

(3) 将 位 0 到 位 c1 - 1 置 1。 

步骤 1、2 有 一 种 快速 做 法 ,将 拖 尾 0 置 为 1 ( 得 到 p 个 拖 尾 1 )， 然 后 再 加 1。 加 1 后 ， 所 有 拖 尾 1 
都 会 翻转 ， 最 终 位 p 变 为 1， 后 面 跟 p 个 0。 我 们 可 以 用 算术 方法 完成 这 些 步骤。 
































十 


n 一 
n += 


2c” - 1;  // 将 拖 尾 8 置 1， 得 到 p 个 拖 尾 1 
13 // 先 将 p 个 1 清 索 ， 然 后 位 p 改 为 1 


接着 ， 用 算术 方法 执行 步 又 3， 如 下 : 
n += 2 ， - 1; // 将 拖 尾 的 c1 - 1 个 8 置 为 1 
上 面 的 数学 运算 可 缩减 为 : 


n+ (2 -1)+1+(2 
n+2®”+2" .1-1 


next 0 


这 种 做 法 的 精妙 之 处 在 于 ， 只 需 一 两 个 位 操作 ， 代 码 写 起 来 也 很 简单 。 


1 int getNextArith(int n) { 

2 /* 计算 c9 和 c1， 跟 之 前 一 样 */ 

3 return n+ (1 << cc0)+(1x< (cl -1)) -1; 
4 小 


5. 算术 解法 ， 获取 前 一 个 数 

如 果 c1 是 拖 尾 1 的 个 数 ，ce8 是 拖 尾 1 右 方 全 为 0 的 位 的 个 数 ， 则 p = ce@ + c1， 前 面 的 getPrev 
可 以 重新 表述 如 下 。 

(1) 将 位 p 清 零 。 
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(2) 将 位 p 右 边 的 所 有 位 置 1。 
(3) 将 位 0 到 位 ce - 1 清 零 。 
上 述 步 又 用 算术 方法 实现 如 下 。 为 简化 起 见 ， 这 里 假定 n = 188968611， 故 c1 = 2 目 c@ = 5。 


n -= 2 - 1; // 清除 拖 尾 1，n 变 为 19966668 

n -= 1; // 翻转 拖 尾 8，n 变 为 91111111 

n -= 2 -1-1;  ”// 翻转 最 右边 (C9-1) 个 1，n 变 为 91116868 
由 此 导出 : 

next n - (2° - 1) 1 瑟 (2 可 1) 


n-29-2 -7+1 


和 getNextArith 一 样 ， 实 现 起 来 很 简单 : 


1 int getPrevArith(int n) { 

2 /* 计算 c9 和 c1， 跟 之 前 一 样 */ 

3 return n - (1<<c1l)- (1x< (ce -1)) +1; 
4 } 





哟 ! 别 紧 张 ， 在 面试 中 ， 你 用 不 着 写 出 上 面 所 有 解法 ， 至 少 不 会 是 在 没有 面试 官 的 大 力 帮 助 
之 :gs 

5.4 解释 代码 ((n & (n-1)) == 9) 的 具体 含义 。( 第 56 页 ) 

解法 

我 们 可 以 由 外 而 内 来 解决 这 个 问题 。 

1. (A & B) == 0 是 什么 意思 ? 

意思 是 ，A 和 B 二 进 制 表示 的 同一 位 置 绝 不 会 同时 为 1。 因 此 ， 如 果 (n & (n-1)) == 6， 则 n 
和 n-1 就 不 会 有 共同 的 1。 


2. 相 比 n，n-1 长 什么 样 ? 
试 着 动手 做 一 下 减法 (二进制 或 十 进 制 )， 结 果 会 怎么 样 ? 























1161611666 [base 2] 593166 [base 16] 
s 1 = 1 
= 1161616111 [base 2] = 593699 [base 16] 





当 要 将 一 个 数 减 去 1 时 ， 需 要 注意 最 低 有 效 位 。 如 果 最 低 有 效 位 为 1， 则 变 为 0， 完 毕 。 如 果 
是 0， 你 就 必须 从 高 位 “ 借 ”1。 因 此 ， 要 逐一 前 往 更 高 的 位 ， 将 每 个 位 从 0 改 为 1， 直 至 找到 1 为 
止 ， 并 将 这 个 1 翻转 成 0， 完 毕 。 

综 上 ，n-1 会 很 像 n， 只 不 过 n 中 低位 的 0 在 n-1 中 变 为 1，n 中 最 低 有 效 位 的 1 在 n-1 中 变 为 0， 
示例 如 下 : 

if n 

then n-1 




















abcde1666 
abcde6111 


那么 ，(n & (n-1)) == 0 究竟 表示 什么 ? 
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n 和 n-1 不 存在 同一 位 均 为 1 的 情况 ， 因 为 两 者 的 二 进 制 表示 如 下 : 


if n = abcde1666 
then n-1 = abcde6111 


abcde 必 定 全 为 0， 也 就 是 说 ，n 必 须 像 是 eeee16886， 因 此 ，n 的 值 是 2 的 某 次 方 。 

综 上 ， 这 个 问题 的 答案 为 : ((n& (n-1)) == 68) 检查 n 是 否 为 2 的 某 次 方 〈 或 者 检查 n 是 否 为 0 )。 
5.5 ”编写 一 个 函数 ， 确 定 需要 改变 几 个 位 ， 才 能 将 整数 A 转 成 整数 B。( 第 57 页 ) 

解法 

这 个 问题 看 似 复杂 , 实则 非常 简单 。 要 解决 这 个 问题 ， 就 得 设法 找 出 两 个 数 之 间 有 哪些 位 不 

















同 


很 简单 ， 使 用 异 或 (XOR ) 操作 即 可 。 
在 异 或 操作 的 结果 中 ， 每 个 1 代表 4 和 8B 相应 位 是 不 一 样 的。 因此 ,要 找 出 4 和 B 有 多 少 个 不 同 





的 位 ， 只 要 数 一 数 4^B 有 几 个 位 为 1。 


























1 int bitSwapRequired(int a, int b) { 

2 int count = 0; 

3 for (int C=a^ 人 bicl=6jc=c>>1) 1{ 

4 Count += Cc &1; 

5 } 

6 return count; 

7 3} 

上 面 的 代码 已 经 很 不 错 了 ， 不 过 还 可 以 做 得 更 好 。 上 面 的 做 法 是 不 断 对 c 执 行 移 位 操作 ， 然 


后 检查 最 低 有 效 位 ， 但 其 实 可 以 不 断 翻 转 最 低 有 效 位 ， 计 算 要 多 少 次 c 才 会 变 成 0。 操 作 c = c & 


(c - 1) 会 清除 c 的 最 低 有 效 位 。 


下 面 的 代码 运用 了 这 个 方法 。 


1 public static int bitSwapRequired(int a, int b) { 
2 int count = 0@; 

3 for (int c=a^b;c!=060;c=c&(c-1))t{ 

4 Count++; 

5 } 

6 return count; 


2 
这 段 代 码 是 偶尔 会 在 面试 中 出 现 的 位 操作 问题 。 如 果 之 前 从 未 见 过 , 一 时 很 难 在 面试 现场 想 





出 来 ， 记 住 这 个 技巧 ， 对 面试 会 很 有 帮助 。 


5.6 ”编写 程序 ， 交 换 某 个 整数 的 奇数 位 和 偶数 位 ， 使 用 指令 越 少 越 好 ( 也 就 是 说 , 位 0 与 


位 1 交换 , 位 2 与 位 3 交换 , 依 此 类 推 ) (第 57 页 ) 


解法 
跟 之 前 儿 个 问题 一 样 ， 从 不 同 角 度 考虑 这 个 问题 会 很 有 帮助 。 要 操作 一 对 一 对 的 位 ,必定 困 


难 重 重 ， 效率 也 不 见得 会 高 。 那 么 ,还 有 其 他 什么 方式 来 解决 这 个 问题 ? 


我 们 可 以 这 么 做 : 先 操作 奇数 位 ,然后 再 操作 偶数 位 。 有 办 法 将 数字 n 的 奇数 位 左 移 或 右 移 1 


位 吗 ? 当然 有 。 我 们 可 以 用 16161618 ( 即 exAA ) 作为 掩 码 ， 提 取 奇 数位 ， 并 将 它们 右 移 1 位 ， 移 
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到 偶数 位 的 位 置 。 对 于 偶数 位 ， 可 以 施 以 同样 的 操作 。 最 后 ， 将 两 次 操作 的 结果 合并 成 一 个 值 。 
这 种 做 法 共 需 5 条 指令 ， 实 现代 码 如 下 。 
1 public int swapOddEvenBits(int x) { 


2 return ( ((x & 6xaaaaaaaa) >> 1) | ((x & @x55555555) << 1) ); 
3 } 


上 述 Java 代 码 实 现 的 是 32 位 整数 。 如 和 欲 处 理 64 位 整数 ， 那 就 需要 修改 掩 码 。 不 过 ， 人 处 理 逻 辑 
是 一 样 的 。 


5.7 ”数组 A 包 售 0 到 n 的 所 有 整数 ， 但 其 中 缺 了 一 个 。 在 这 个 问题 中 ， 只 用 一 次 操作 无 法 取 
得 数组 A 里 某 个 整数 的 完整 内 容 。 此 外 ， 数 组 A 的 元 素 皆 以 二 进 制 表示 ， 唯 一 可 用 的 访问 操作 是 
“从 A[i] 取 出 第 /位 数据 ”， 该 操作 的 时 间 复 杂 度 为 常数 。 请 编写 代码 找 出 那个 缺失 的 整数 。 你 有 
办 法 在 O(n) 时 间 内 完成 吗 ? 〈 第 57 页 ) 


解法 

你 可 能 听 到 过 非常 类 似 的 问题 : 给 定 一 列 0 到 xz 的 数 ， 其 中 只 缺 一 个 数字 ， 把 这 个 数 找 出 来 。 
这 个 问题 解决 起 来 很 简单 ， 直 接 将 这 列 数 相 加 ， 然 后 与 0 到 xz 的 总 和 ( 即 n* (n+1)/2) 进行 比较 。 
两 者 差 值 就 是 那个 缺失 的 整数 。 

至 于 这 一 题 ， 我 们 可 以 根据 每 个 整数 的 二 进 制 表示 ， 求 出 它 的 值 ， 然 后 计算 总 和 。 

这 种 解法 的 执行 时 间 为 n * length(n)， 其 中 length 为 中 有 和 多少 个 位 。 注意, length(n)= 1og2(n)， 
因此 ， 真 正 的 执行 时 间 为 O(n log(n))， 效 率 不 够 高 ! 那么 ， 我 们 该 怎么 办 呢 ? 

其 实 ， 我 们 可 以 使 用 类 似 的 解法 ， 不 过 会 更 直接 第 利用 每 个 位 的 值 。 

假设 有 下 面 这 些 二 进 制 数 ( ----- 表示 移 除 的 那个 数 ): 








六 























66666 66166 91666 61166 
66661 66161 691661 61161 
66616 66116 91616 
de 686111 810611 


移 除 上 面 那个 数 会 导致 最 低 有 效 位 ( 记 作 LSB; ) 中 1 和 0 的 失衡 。 在 0 到 n 的 数 中 ， 若 n 为 奇数 ， 
则 0 和 1 的 数量 相同 ; 若 n 为 偶数 ， 则 0 比 1 的 数量 多 一 个 ， 也 就 是 说 : 
若 n % 2 
若 n % 2 
由 此 可 见 ，count(8s) 必 定 大 于 或 等 于 count(1s)。 
从 这 列 数 中 移 除数 值 vy 后 , 只 要 检查 其 他 数值 的 最 低 有 效 位 , 马上 就 能 知道 "是 偶数 ; 


1， 则 count(6s) = count(1s) 
8， 则 count(6s) = 1 + count(1s) 


Ee 
[ou 
己 
泛 





n%2==6 1 re 
count(6s) = 1 + count(1s) count(6s) = count(1s) 

















vV%2 ==0 a 0 is removed. a @ is removed. 
LSB(V) = 6 | count(6s) = count(1s) count(6s) < count(1s) 
TS 汗 a 1 is removed. a 1 is removed. 
LSB,(vV) = 1 | count(6s) > count(1s) Count(6s) > count(1s) 
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因此 ， 如 果 count(8s) <= count(1s)， 则 * 为 偶数 ， 如 果 count(8s) > count(1s)， 则 v* 为 奇数 。 

那么 ,我 们 又 该 如 何 确定 v 的 下 一 个 位 呢 ? 如 果 这 列 数 中 含有 * 的 话 ,我 们 就 会 发 现 如 下 规律 
《其 中 countb 表 示 第 二 个 最 低 有 效 位 中 0 或 1 的 个 数 ): 

countz(0s) = countz(1s) 或 countz(0s) = 1 + countz(1s) 


跟前 面 的 例子 一 样 ， 我 们 可 以 推导 出 * 的 第 二 个 最 低 有 效 位 【LSB， )。 





| count;(6s) = 1 + count,(1s) | count, (0s) = count,(1s) | 





LSB,(V) == a 8 is removed. a 8 is removed. 
count,(8s) = count,(1s) count,(8s) < count,(1s) 


LSB,(v) == 1 |a 1 is removed. a 1 is removed. 
count,(6s) > count,(1s) count,(6s) > count,(1s) 


同样 的 ， 我 们 可 以 得 出 以 下 结论 : 
口 若 count(0s) <= counts(1s)， 则 LSB2(v) = 0。 
口 知 countz(0s) > counts(1s)， 则 LSB2(V) = 1。 
重复 上 述 操 作 可 以 找 出 每 个 位 ,每 次 迭代 时 ,我 们 数 出 位 ;中 0 和 1 的 数量 ,检查 LSB/V) 是 0 还 是 
1。 然 后 ， 气 弃 LSB; (x) != LSB; () 的 那些 数字 。 也 就 是 说 ， 若 ,为 偶数 ， 则 据 弃 奇数 ， 依 此 类 推 。 
在 操作 流程 的 最 后 ， 就 可 得 到 v 所 有 位 的 值 。 在 每 一 次 迭代 中 ,我 们 会 查看 n 个 位 ， 然 后 是 n/ 
2 个 ， 接 着 是 n / 4 个 ， 等 等 。 因 此 ， 时 间 复 杂 度 为 O(N)。 
我 们 还 可 以 更 形象 地 演示 整个 过 程 。 在 第 一 次 迭代 时 ， 有 下 面 这 些 数字 : 


















































00660 60166 1066 60116066 
606061 066161 68016861 81161 
09606016 96116 916016 
----- 86680111 81611 
由 counti(0s) > counti(1s) 可 知 LSB1(v) = 1。 因 此 ， 握 除 所 有 使 得 LSB1(x) != LSB1(v) 的 数 x。 
066668 961668 861686668 t+668 
96060661 96161 81661 91161 
096616 80+1+8 91616 
----- 8668111 6810611 
接着 ， 由 countz(0s) > counts(1s) 可 知 LSB2(v) = 1。 因 此 ， 握 除 所 有 使 得 LSB2(x) != LSB20Y) 的 数 x。 
0888868 06166 60168666 +166 
6666 B616+ 8+080+ 8++06+ 
096616 061+t6 8+08+8 
----- 8696111 91611 
此 时 ， 由 counts(0s) <= counts(1s) 可 知 LSBs3(v)=0。 因 此 ， 气 除 所 有 使 得 LSB3Q) (= LSB30W) 的 数 x。 
066668 08686166 14168666 1166 
6666 80+0+ 8+80+ 8++06+ 
096616 0861+16 3191 
----- 808+++ 91611 


最 后 只 剩 下 一 个 数字 了 ， 此 时 count4(0s) <= counts(1s)， 因 此 LSB4(V) = 0。 
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气 除 所 有 使 得 LSB4(v) != 0 的 数字 之 后 ， 我 们 得 到 一 个 空 的 列表 。 列 表 为 空 之 后 ， 可 以 得 到 

count;(0s) <= count(1s)， 因 此 LSBi( = 0。 换 句 话 说 ,一旦 列表 为 空 ， 即 可 将 vy 的 其 余 位 填 为 0。 
在 上 面 的 示例 中 ， 整 个 操作 流程 将 算出 v = 88811。 
下 面 是 该 算法 的 实现 代码 ， 我 们 按 位 值 切 分 整个 数组 ， 借 此 实现 了 气 除 部 分 代码 。 





1 public int findMissing(ArrayList<BitInteger> array) { 
2 /* bit @ 对 应 于 LSB。 以 此 为 起 点 ， 

3 * 逐步 向 较 高 的 位 推进 */ 

4 return findMissing(array, 0); 

县 站 

6 

7 public int findMissing(ArrayList<BitInteger> input, int column) { 
8 if (column >= BitInteger.INTEGER_SIZE) { // 终止 条 件 与 错误 条 件 
9 return 0@; 

16 } 

11 ArrayList<BitInteger> oneBits = 

12 new ArrayList<BitInteger>(input.size()/2); 

13 ArrayList<BitInteger> zeroBits = 

14 new ArrayList<BitInteger>(input.size()/2); 

15 

16 for (BitInteger t : input) { 

17 if (t.fetch(column) == 9) { 

18 zeroBits.add(t); 

19 } else { 

26 oneBits.add(t); 

了 } 

22 

23 if (zeroBits.size() <= oneBits.size()) { 

24 int v = findMissing(zeroBits, column + 1); 

25 return (v << 1) | @; 

26 } else { 

27 int v = findMissing(oneBits, column + 1); 

28 return (v << 1) | 1; 

29 } 

36 } 











在 第 24 和 27 行 ， 我 们 会 以 递归 方式 计算 出 y 的 其 他 位 。 然 后 ， 再 根据 counti(0s) <= counti(1s) 
是 否 成 立 ， 插入 0 或 1。 


5.8 有 个 单 色 屏幕 存储 在 一 个 一 维 字 节 数组 中 ， 使 得 8 个 连续 像素 可 以 存放 在 一 个 字 节 
里 。 屏 幕 宽度 为 w， 且 w 可 被 8 整除 ( 即 一 个 字 节 不 会 分 布 在 两 行 上 )， 屏 幕 高 度 可 由 数组 长 度 及 
屏幕 宽 度 推算 得 出 。 请 实现 一 个 函数 drawHorizontalLine(byte[] screen, int width，int x1， 
int x2，int y)， 绘 制 从 点 (Xx1, y) 到 点 (xz, y) 的 水 平 线 。( 第 57 页 ) 


解法 

这 个 问题 有 个 粗糙 的 简单 解法 : 用 for 循 环 人 迭代， 从 x1 到 x2， 一 路 设 定 每 个 像素 。 但 这 么 做 
太 没 劲 了 ， 是 吧 ? (况且 效率 也 不 高 。) 

更 好 的 做 法 是 , 如 果 x1 和 x2 相 距 甚 远 , 其 间 包 含 几 个 完整 字 节 。 只 要 使 用 screen[byte_pos] 
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1 
这 
[| 
i 
一 
nn 





= 6xFF， 一 次 就 能 设 定 一 整个 字 节 。 这 条 线 起 点 和 终点 剩余 部 分 的 位 ， 可 用 掩 码 设 定 。 





1 void drawLine(byte[] screen, int width, int x1, int x2, int y) { 
之 int start offset = x1 % 8; 

3 int first full_ byte = x1 / 8; 

4 if (start offset != 0) { 

5 first full_ bytet+; 

6 } 

学 

8 


int end_offset = x2 % 8; 


9 int last full byte = x2 / 8; 
16 if (end_offset != 7) { 

11 last full_byte--; 

12 } 

13 


14  ”// 设 定 完整 的 字 节 

15 for (int b = first full byte; b <= last full byte; b++) { 
16 screen[(width / 8) * y + b] = (byte) 6xFF; 

17 } 


19 // 创建 用 于 线条 起 点 和 终点 的 掩 码 
26 byte start mask = (byte) (9xFF >> start_offset ) ; 
21 byte end mask = (byte) ~(6xFF >> (end_offset + 1)); 


23 // 设 定 线条 的 起 点 和 终点 
24 if ((x1 / 8) == (x2 / 8)) { // x1 和 Xx2 位 于 同一 字 节 


25 byte mask = (byte) (start mask & end_mask); 

26 screen[(width / 8) + y + (x1 / 8)] |= mask; 

27 } else { 

28 if (start offset != 6) { 

29 int byte number = (width / 8) * y + first full byte - 1; 
36 screen[byte_number] |= start mask; 

31 } 

32 if (end offset != 7) { 

33 int byte number = (width / 8) * y + last full byte + 1; 
34 screen[byte number] |= end_mask; 

35 } 

36 } 

37 } 


务必 小 心 处 理 这 个 问题 ， 其 中 暗藏 许多 “陷阱 ”和 特殊 情况 。 例 如 ， 你 必须 考虑 到 x1 和 x2 
位 于 同一 字 节 的 情况 。 只 有 那些 最 细心 的 求职 者 ， 才 能 毫 无 丝 漏 地 写 出 这 段 代 码 。 


9.6 智力 题 


6.1 有 20 瓶 药丸 ， 其 中 19 瓶装 有 1 克 / 粒 的 药丸 ， 余 下 一 瓶装 有 1.1 克 / 粒 的 药丸 。 给 你 
一 台 称 重 精准 的 天 平 ， 怎 么 找 出 比较 重 的 那 瓶 药丸 9” 天 平 只 能 用 一 次 。( 第 59 页) 





解法 
有 时 候 , 严格 的 限制 条 件 有 可 能 反倒 是 解 题 的 线索 。 在 这 个 问题 中 ,限制 条 
一 次 。 
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因为 天 平 只 能 用 一 次 , 我 们 也 得 以 知道 一 个 有 趣 的 事实 : 一 次 必须 同时 称 很 多 药丸 ,其实 更 
准确 地 说 ， 是 必须 从 19 瓶 拿 出 药丸 进行 称 重 。 和 否则 ， 如 果 跳 过 两 瓶 或 更 多 瓶 药丸， 又 该 如 何 区 分 
没 称 过 的 那 几 瓶 呢 ? 别 忘 了 ， 天 平 只 能 用 一 次 。 

那么 , 该 怎么 称 重 取 自 多 个 药 瓶 的 药丸 ,并 确定 哪 一 瓶装 有 比较 重 的 药丸 ”假设 只 有 两 瓶 药 
丸 ， 其 中 一 瓶 的 药丸 比较 重 。 每 瓶 取 出 一 粒 药丸 ， 称 得 重量 为 2.1 克 ， 但 无 从 知道 这 多 出 来 的 0.1 
克 来 自 哪 一 瓶 。 我 们 必须 设法 区 分 这 些 药 瓶 。 

如 果 从 药 瓶 所 取出 一 粒 药 丸 ， 从 药 瓶 检 取 出 两 粒 药丸 , 那么, 称 得 重量 为 多 少 呢 ? 结果 要 看 
情况 而 定 。 如 果 药 瓶 所 的 药丸 较 重 ， 则 称 得 重量 为 3.1 克 。 如 果 药 瓶 儿 的 药丸 较 重 ， 则 称 得 重量 
为 3.2 克 。 这 就 是 这 个 问题 的 解 题 窍门 。 

称 一 堆 药 九 时 ,我 们 会 有 个 “预期 ” 重量。 而 借 由 预期 重量 和 实测 重量 之 间 的 差别 ， 就 能 得 
出 哪 一 瓶 药 丸 比较 重 ， 前 提 是 从 每 个 药 瓶 取出 不 同 数量 的 药丸 。 

将 之 前 两 瓶 药 丸 的 解法 加 以 推广 ， 就 能 得 到 完整 解法 . 从 药 瓶 抽取 出 一 粒 药丸 ， 从 药 瓶 要 
取出 两 粒 ， 从 药 瓶 妇 取 出 三 粒 ， 依 此 类 推 。 如 果 每 粒 药丸 均 重 1 克 ， 则 称 得 总 重量 为 210 克 (1+ 2 
+ +20 = 20 * 21 / 2 = 210),“ 多 出 来 的 ”重量 必定 来 自 每 粒 多 0.1 克 的 药丸 。 

药 瓶 的 编号 可 由 算式 (weight - 216 grams) / 6.1 grams 得 出 。 因 此 ， 若 这 堆 药 丸 称 得 重 
量 为 211.3 克 ， 则 药 瓶 #13 装 有 较 重 的 药丸 。 


6.2 有 个 8x8 棋盘 ， 其 中 对 角 的 角落 上 ， 两 个 方 格 被 切 掉 了 。 给 定 31 块 多 米 诺 骨牌 ， 
一 块 骨 牌 恰好 可 以 履 盖 两 个 方 格 。 用 这 31 块 骨牌 能 否 盖 住 整个 棋盘 9 请 证 明 你 的 答案 〈 提供 范 
例 ， 或 证 明 为 什么 不 可 能 )。( 第 59 页 ) 


解法 

乍 一 看 ， 似 乎 是 可 以 盖 住 的 。 棋 盘 大 小 为 8x8， 共 有 64 个 方 格 ， 但 其 中 两 个 方 格 已 被 切 掉 ， 
因此 只 剩 62 个 方 格 。31 块 骨牌 应 该 刚好 能 盖 住 整个 棋盘 ， 对 吧 ? 

答 试 用 骨牌 盖 住 第 1 行 ， 而 第 1 行 只 有 7 个 方 格 ,因此 有 一 块 骨牌 必须 铺 至 第 2 行 。 而 用 骨牌 盖 
住 第 2 行 时 ， 我 们 又 必须 将 一 块 骨牌 铺 至 第 3 行 。 


















































































































要 盖 住 每 一 行 ， 总 有 一 块 骨 牌 必须 铺 至 下 一 行 。 无 论 尝试 多 少 次 、 多 少 种 方法 ,我 们 都 无 法 
成 功 铺 下 所 有 骨牌 。 
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其 实 , 还 有 更 简洁 更 严谨 的 证 明说 明 为 什么 不 可 能 。 棋盘 原本 有 32 个 黑 格 和 32 个 白 格 。 将 对 
角 角 沙 上 的 两 个 方 格 ( 相同 颜色 ) 切 掉 ， 棋 盘 只 剩 下 30 个 同色 的 方 格 和 32 个 另 一 种 颜色 的 方 格 。 
为 方便 论证 起 见 ， 我 们 假定 棋盘 上 剩 下 30 个 黑 格 和 32 个 白 格 。 

放 在 棋盘 上 的 每 块 骨牌 必定 会 盖 住 一 个 白 格 和 一 个 黑 格 。 因 此 , 31 块 骨牌 正好 盖 住 31 个 白 格 
和 31 个 黑 格 。 然 而 ， 这 个 棋盘 只 有 30 个 黑 格 和 32 个 白 格 ， 所 以 ，31 块 骨牌 盖 不 住 整个 棋盘 。 


6.3 ”有 两 个 水 壹 ， 容 量 分 别 为 5 夸 脱 (美制 : 1 夸 脱 =0.946 升 ， 英 制 : 1 夸 脱 =1.136 升 ) 
和 3 夸 脱 ， 若 水 的 供应 不 限量 (但 没有 量 杯 )， 怎 么 用 这 两 个 水 过 得 到 刚好 4 夸 脱 的 水 9 注意 ， 
这 两 个 水 毒 呈 不 规则 形状 ， 无 法 精准 地 装 满 “ 半 壶 ”水 。( 第 59 页 ) 

解法 

根据 题 意 , 我 们 只 能 使 用 这 两 个 水 过 ,不 妨 随 意 把 玩 一 番 ， 把 水 倒 来 倒 去 ,可 以 得 到 如 下 顺 
序 组 合 : 

























































































































































































5 夸 脱 3 夸 脱 注解 
3 0 装 靖 5 和 对 脱水 姓 
2 3 用 5$ 夸 脱水 壶 里 的 水 装 满 3 压 脱 水 豆 
0 2 把 $ 夸 脱水 壹 里 的 水 倒 人 3 夸 脱 水 喜 
5 2 装 并 $ 夸 脱水 过 
4 3 用 5$ 夸 脱水 壹 里 的 水 填 满 3 夸 脱 水 壹 
4 搞定 ! 准确 量 得 4 夸 脱 。 

注意 , 许多 智力 题 其 实 都 隐 含 数学 或 计算 机 科学 的 背景 ， 这 个 问题 也 不 例外 。 只 要 这 两 个 水 





注意 
壶 的 容量 互 质 ( 即 两 个 数 没有 共同 的 质 因 子 )， 我 们 就 能 找 出 一 种 倒 水 的 顺序 组 合 ， 量 出 1 到 2 个 
水 壶 容量 总 和 ( 含 ) 之 间 的 任意 水 量 。 


6.4 ”有 个 岛 上 住 着 一 群 人 ， 有 一 天 来 了 个 游客 ， 定 了 一 条 奇怪 的 规矩 : 所 有 蓝 眼 睛 的 人 都 
必须 尽快 离开 这 个 岛 。 每 晚 8 点 会 有 一 个 航班 离岛 。 每 个 人 都 看 得 见 别 人 眼睛 的 颜色 , 但 不 知道 
自己 的 《别人 也 不 可 以 告知 )。 此 外 ， 他 们 不 知道 岛 上 到 底 有 多 少 人 是 蓝 眼睛 的 ， 只 知道 至 少 有 
一 个 人 的 眼睛 是 蓝 色 的 。 所 有 蓝 眼睛 的 人 要 花 几 天 才能 离开 这 个 岛 ?* 〈 第 59 页 ) 


解法 
下 面 将 采用 简单 构造 法 。 假 定 这 个 品 上 一 共有 人， 其 中 c 人 有 蓝 眼 睛 。 由 题目 可 知 ，c> 0。 


= 
| 


2 





1. 情况 c= 1: 只 有 一 人 是 蓝 眼睛 的 
假设 岛 上 所 有 人 都 是 聪明 的 ， 蓝 眼睛 的 人 四 人 处 观察 之 后 ,发 现 没 有 人 是 蓝 眼 睛 的 。 但 他 知道 
至 少 有 一 人 是 蓝 眼睛 的 , 于 是 就 能 推导 出 自己 一 定 是 蓝 眼睛 的 。 因此 , 他 会 搭乘 当晚 的 飞机 离开 。 



































2. 情况 c = 2: 只 有 两 人 是 蓝 眼 睛 的 
两 个 蓝 眼 睛 的 人 看 到 对 方 ， 并 不 确定 c 是 1 还 是 2， 但 是 由 上 一 种 情况 ， 他 们 知道 ， 如 果 c =1， 9 
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那个 蓝 眼 睛 的 人 第 一 晚 就 会 离岛 。 因此 , 发 现 男 一 个 蓝 眼睛 的 人 仍 在 岛 上 , 他 一 定 能 推断 出 c= 2， 
也 就 意味 着 他 自己 也 是 蓝 眼 睛 的 。 于 是 ， 两 个 蓝 眼睛 的 人 都 会 在 第 二 晚 离岛 。 


3. 情况 c > 2: 一 般 情 况 

逐步 提高 ce 时 ， 我 们 可 以 看 出 上 述 逻 辑 仍 旧 适 用 。 如 果 c = 3， 那 么 ， 这 三 个 人 会 立即 意识 到 有 2 
到 3 人 是 蓝 眼睛 的 。 如 果 有 两 人 是 蓝 眼睛 的 ， 那么 这 两 人 会 在 第 二 晚 离岛 。 因 此 ， 如 果 过 了 第 二 晚 另 
外 两 人 还 在 名 上 ， 每 个 蓝 眼 睛 的 人 都 能 推断 出 c=3 ， 因 此 这 三 人 都 有 蓝 眼 睛 。 他 们 会 在 第 三 晚 离 品 。 

不 论 c 为 什么 值 ， 都 可 以 套用 这 个 模式 。 所 以 ， 如 果 有 c 人 是 蓝 眼 睛 的 ， 则 所 有 蓝 眼 睛 的 人 要 
用 c 晚 才能 离岛 ， 且 都 在 同一 晚 离 开 。 


6.5 ”有 栋 建 筑 物 高 100 层 。 若 从 第 NN 层 或 更 高 的 楼 层 扔 下 来 ， 鸡 蛋 就 会 破 掉 。 若 从 第 NN 层 
以 下 的 楼 层 扔 下 来 则 不 会 破 掉 。 给 你 2 个 鸡蛋 , 请 找 出 N, 并 要 求 最 差 情况 下 扔 鸡蛋 的 次 数 为 最 
少 。( 第 59 页 ) 


解法 

我 们 发 现 ， 无 论 怎 么 扔 鸡蛋 1 ( Egg 1 )， 鸡 蛋 2 ( Egg 2 ) 都 必须 在 “ 破 掉 那 一 层 ” 和 下 一 个 不 
会 破 掉 的 最 高 楼 层 之 间 ， 逐 层 扔 下 楼 ( 从 最 低 的 到 最 高 的 )。 例 如 ， 若 鸡蛋 1 从 $ 层 和 10 层 楼 扔 下 没 
破 掉 ， 但 从 15 层 扔 下 时 破 掉 了 ,那么 ,在 最 差 情况 下 ， 鸡 蛋 2 必须 尝试 从 11、12 、13 和 14 层 扔 下 楼 。 


具体 做 法 
首先 ， 让 我 们 试 着 从 10 层 开始 扔 鸡蛋 ， 然 后 是 20 层 ， 等 等 。 
口 如 果 鸡 蛋 1 第 一 次 扔 下 楼 ( 10 层 ) 就 破 掉 了 ,那么 ， 最 多 需要 扔 10 次 。 
口 如 果 鸡 和 蛋 1 最 后 一 次 扔 下 楼 (100 层 ) 才 破 掉 ， 那 么 ， 最 多 要 扔 19 次 (10、20、…、90、 
100 层 ， 然 后 是 91 到 99 层 )。 

这 么 做 也 挺 不 错 ， 但 我 们 只 考虑 了 绝对 最 差 情况 。 我 们 应 该 进行 “负载 均衡 *"， 让 这 两 种 情 
况 下 扔 鸡蛋 的 次 数 更 均匀 。 

我 们 的 目标 是 设计 一 种 扔 鸡蛋 的 方法 ， 使 得 扔 鸡蛋 1 时 ， 不 论 是 在 第 一 次 还 是 最 后 一 次 扔 下 
楼 才 破 掉 ， 次 数 越 稳 定 越 好 。 

(1) 完美 负载 均衡 的 方法 应 该 是 , 扔 鸡蛋 1 的 次 数 加 上 扎 鸡 恒 2 的 次 数 ， 不论 什么 时 候 都 一 样 ， 
不 管 鸡蛋 1 是 从 哪 层 楼 扔 下 时 破 掉 的 。 

(2) 若 有 这 种 扔 法 ， 每 次 鸡蛋 1 多 扔 一 次 ， 鸡 蛋 2 就 可 以 少 扔 一 次 。 

(3) 因此 ， 每 丢 一 次 鸡蛋 1， 就 应 该 减少 鸡蛋 2 可 能 需要 扔 下 楼 的 次 数 。 例 如 ， 如 果 鸡 蛋 1 先 从 
20 层 往 下 扔 ， 然 后 从 30 层 扔 下 楼 ， 此 时 鸡蛋 2 可 能 就 要 扔 9 次 。 若 鸡蛋 1 再 扔 一 次 ， 我 们 必须 让 鸡 
蛋 2 扔 下 楼 的 次 数 降 为 8 次 。 也 就 是 说 ， 我 们 必须 让 鸡蛋 1 从 39 层 扔 下 楼 。 

(4) 由 此 可 知 ， 鸡 蛋 1 必 须 从 X 层 开始 往 下 护 ， 然 后 再 往 上 增加 1 层 …… 直 至 到 达 100 层 。 

(5) 求解 方程 式 Y+ (和 DJ + (X-2) +… +1=100, 得 到 X(X+1)/2=100 一 XY=14。 

我 们 先 从 14 层 开始 ， 然 后 是 27 层 ， 接 着 是 39 层 ， 依 此 类 推 , 最 差 情况 下 鸡蛋 要 扔 14 次 。 

正如 解决 其 他 许多 最 大 化 /最 小 化 的 问题 一 样 ， 这 类 问题 的 关键 在 于 “平衡 最 差 情况 ”。 
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6.6 ”走廊 上 有 100 个 关上 的 储 物 柜 。 有 个 人 先是 将 100 个 柜子 全 都 打开 。 接 着 ， 每 数 两 个 
柜子 关上 一 个 。 然 后, 在 第 三 轮 时 ， 表 每 隔 两 个 就 切换 第 三 个 柜子 的 开关 状态 ( 也 就 是 将 关上 的 
柜子 打开 ， 将 打开 的 关上 )。 照 此 规律 反复 操作 100 次 ， 在 第 轮 ， 这 个 人 会 每 数 / 个 就 切换 第 /个 
柜子 的 状态 。 当 第 100 轮 经 过 走廊 时 ， 只 切换 第 100 个 柜子 的 开关 状态 ， 此 时 有 几 个 柜子 是 开 
着 的 ? (第 59 页 ) 


解法 
要 解决 这 个 问题 , 我 们 必须 弄 清 楚 所 谓 切 换 储 物 柜 开关 状态 是 什么 意思 。 这 有 助 于 我 们 推断 
最 终 哪 些 柜子 是 开 着 的 。 


1. 问题 : 柜子 会 在 哪 几 轮 切换 状态 〈 开 或 关 ) ? 
柜子 n 会 在 n 的 每 个 因子 (包括 1 和 xz 本 身 ) 对 应 的 那 一 轮 切 换 状 态 。 也 就 是 说 ， 柜 子 15 会 在 第 
1、3、5 和 15 轮 开 或 关 一 次 。 


2. 问题 : 柜子 什么 时 候 还 是 开 着 的 ? 
如 果 因 子 个 数 〈 记 作 x ) 为 奇数 ， 则 这 个 柜子 是 开 着 的 。 你 可 以 把 一 对 因子 比 作 开 和 关 ， 若 
还 剩 一 个 因子 ， 则 柜子 就 是 开 着 的 。 


3. 问题 : X 什 么 时 候 为 奇数 ? 

若 n 为 完全 平方 数 ， 则 x 的 值 为 奇数 。 理 由 如 下 : 将 n 的 两 个 互补 因子 配对 。 例 如 ， 如 7 为 36， 
则 因子 配对 情况 为 : (1, 36)、(2, 18)、(3, 12)、(4, 9)、(6, 6)。 注 意 ，(6, 6) 其 实 只 有 一 个 因子 ， 
此 n 的 因子 个 数 为 奇数 。 


4. 问题 : 有 多 少 个 完全 平方 数 ? 

一 共有 10 个 完全 平方 数 ， 你 可 以 数 一 数 (1、4、9、16、25、36、49、64、81 、100 ), 或 者 ， 
直接 列 出 1 到 10 的 平方 : 

1*1，2*x2，3*x3，...，16*x16 


此 ， 最 后 共有 10 个 柜子 是 开 着 的 。 
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7.1 有 个 篮球 框 ， 下 面 两 种 玩法 可 任 选 一 种 。 

玩法 1: 一 次 出 手机 会 ， 投 篮 命 中 得 分 。 

玩法 2: 三 次 出 手机 会 ， 必 须 投 中 两 次 。 

如 果 p 是 某 次 投篮 命中 的 概率 ， 则 p 的 值 为 多 少时 ， 才 会 选择 玩法 1 或 玩法 2? (第 63 页 ) 


解法 
要 解 此 题 ， 我 们 可 以 直接 运用 概率 论 ， 比 较 赢 得 各 种 玩法 的 概率 。 9 
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1. 赢得 玩法 1 的 概率 : 

根据 定义 ， 赢 得 玩法 1 的 概率 为 p。 

2. 赢得 玩法 2 的 概率 ; 

令 s(bm 为 m 次 投篮 准确 投 中 k 次 的 概率 ， 赢 得 玩法 2 的 概率 是 三 投 两 中 或 三 投 三 中 的 概率 。 换 
名 话说: 








P( 获 胜 ) = s(2,3) + s(3,3) 
三 投 三 中 的 概率 为 : 
s(3,3) = 六 
三 投 两 中 的 概率 为 : 
P( 第 1、2 次 投 中 ， 第 3 次 未 投 中 ) 
+P( 第 1、3 次 投 中 ， 第 2 次 未 投 中 ) 
+P( 第 1 次 未 投 中 ， 第 2、3 次 投 中 ) 
=p*p*(l-p)+tp* (lp)*pt+(l-p)*p*p 
=3(1-p)p’ 

两 者 概率 相 加 ， 可 以 得 到 : 
=p +3(1-p)p” 








3. 该 选择 哪 种 玩法 ? 
若 P( 玩 法 1) > P( 玩 法 2)， 则 该 选择 玩法 1: 
p> 3p 2p 
1 > 3p-2p” 
2p—3p+1>0 
(2p-1)(p-1)>0 
左边 两 项 必须 同 为 正 数 或 同 为 负数 。 显 然 , p < 1， 故 p-1 <0， 也 即 这 两 项 必须 同 为 负数 。 
2p-1<0 
2p<1 
p<.5 
综 上 ， 若 p <0.5， 则 应 该 选择 玩法 1。 若 p = 0、0.5 或 1， 则 P( 玩 法 1) = P( 玩 法 2)， 选 哪 种 玩法 
都 一 样 ， 因 为 赢得 两 种 玩法 的 概率 相等 。 
7.2 三 角形 的 三 个 顶点 上 各 有 一 只 蚂蚁 。 如 果 蚂 蚁 开始 沿 着 三 角形 的 边 爬 行 ， 两 只 或 三 只 
蚂蚁 撞 在 一 起 的 概率 有 多 大 9 假定 每 只 蚂蚁 会 随机 选 一 个 方向 , 每 个 方向 被 选 到 的 几率 相等 , 而 
三 只 蚂蚁 的 怜 行 速度 相同 。 
类 似 问 题 : 在 "个 顶点 的 多 边 形 上 有 7 只 蚂蚁 ， 求 出 这 些 蚂 蚁 发 生 碰撞 的 概率 。( 第 63 页 ) 
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解法 
当 其 中 两 上 只 蚂蚁 互相 朝 着 对 方 而 行 ， 就 会 发 生 碰撞 。 因 此 ,蚂蚁 不 发 生 碰 撞 的 前 提 是 ,它们 
都 朝 着 同一 方向 候 行 ( 顺 时 针 或 逆 时 针 )。 我 们 可 以 算出 这 种 情况 的 概率 ， 然 后 再 反 推出 问题 的 


答案 


























每 只 蚂 鸯 可 以 朝 两 个 方向 爬行 ， 一 共有 3 只 蚂蚁 ， 它 们 不 发 生 碰 撞 的 概率 位 : 
P( 顺 时 针 ) = (7 
P( 道 时针 ) = (0%) 
(同方 向 ) = (的 ”+(%) =% 
因此 ， 发 生 碰 撞 的 概率 就 是 蚂蚁 不 朝 着 同方 向 息 行 的 概率 : 
P( 碰 撞 )= 1-P( 同 方向 )= 1 (%)=% 
若 要 将 这 个 方法 推广 至 n 个 顶点 的 多 边 形 ， 同 样 的 ， 昭 蚁 也 只 有 以 顺 时 和 针 或 逆 时 针 同 方向 候 
行 才 不 致 相 撞 ,但 总 共有 >" 种 候 行 方式 。 综 上 ， 发 生 碰 撞 的 概率 为 : 
P( 顺 时 针 ) = (%》” 
P( 逆 时 针 ) = (%)” 
P( 同 方向 ) = 20%)"= (4%)" 
P( 碰 撞 ) = 1-P( 同 方向 ) = 1- (4) 


7.3 ”给 定 直角 坐标 系 上 的 两 条 线 ， 确 定 这 两 条 线 会 不 会 相交 。( 第 63 页 ) 


解法 

此 题 有 很 多 不 确定 的 地 方 : 两 条 线 的 格式 是 什么 ? 两 条 线 实 为 同一 条 怎么 处 理 ? 这 些 含糊 不 
清 的 地 方 最 好 跟 面试 官 讨论 一 下 。 
下 面 将 做 出 以 下 假设 : 
口 若 两 条 线 是 相同 的 〈 斜率 和 和 y 轴 截 距 相 等 )， 则 认为 这 两 条 线 相交 ; 
口 我 们 可 以 决定 线 的 数据 结构 。 

两 条 线 阁 不 平行 则 必 相 交 。 因 此 , 要 检查 两 条 线 相交 与 否 , 我 们 只 需 检查 两 者 的 斜率 是 否 相 
同 , 或 是 否 为 同一 条 。 



































实现 代码 如 下 : 

1 public class Line { 

2 static double epsilon = 6.6606661; 

3 public double slope; 

4 public double yintercept; 

5 

6 public Line(double s, double y) { 

7 slope = s; 

8 yintercept = y; 

9 } 

16 

型 public boolean intersect(Line line2) { 

12 return Math.abs(slope - line2.slope) > epsilon || 
13 Math.abs(yintercept - line2.yintercept) < epsilon; 
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14 } 
15 } 


遇 到 这 类 问题 时 ， 务 请 注意 以 下 几 点 。 

口 多 提问 。 此 题 存在 诸多 不 明之 处 ， 多 提问 以 厘清 问题 。 许 多 面试 官 会 故意 提 些 模糊 的 问 

题 ， 考 察 你 是 否 会 说 明 自 己 的 假设 条 件 。 

口 尽量 设计 并 使 用 数据 结构 ， 借 此 展示 你 了 解 并 注重 面向 对 象 设计 。 

口 仔细 考虑 要 怎么 设计 数据 结构 来 表示 一 条 线 。 选 择 多 多 ， 各 有 优 劣 ， 必 须 权 衡 取 舍 。 选 

择 一 种 数据 结构 ， 并 说 明理 由 。 

口 不 要 假设 斜率 和 y 轴 截 距 就 是 整数 。 

口 了 解 浮 点 表示 法 的 限制 。 切 记 不 要 用 == 检 查 浮 点 数 是 否 相 等 ， 而 是 应 该 检查 两 者 差 值 是 
否 小 于 某 个 极 小 值 ( 如 上 面 代码 中 的 epsilon 值 )。 


7.4 ”编写 方法 ， 实 现 整数 的 乘法 、 减 法 和 除法 运算 。 只 人 允许 使 用 加 号 。( 第 63 页 ) 

解法 

此 题 只 允许 使 用 加 号 运算 符 。 对 于 每 个 子 问 题 , 最 好 深入 思考 这 些 运 算 的 本 质 , 或 者 如 何 用 
其 他 运算 表示 加 法 或 已 实现 的 运算 )。 


1. 减法 

怎样 才能 用 加 法 表示 减法 ?这 个 问题 看 起 来 直截了当 ， 运 算 a - 5b 跟 a + (-1)* 5b 是 一 回 事 。 不 
根据 题 意 ， 不 得 使 用 乘 号 (* )， 因 此 我 们 必须 实现 一 个 取 反 ( negate ) 的 函数 。 

1 /* 正 号 变 负 号 ， 负 号 变 正 号 */ 

2 public static int negate(int a) { 

3 int neg = 0) 


































































































4 int d=ax<0?1: -1; 
5 while (a != 6) { 

6 neg += d; 

7 a += d; 

8 } 

9 return neg; 

16 } 

11 


12 /* 两 数 相 减 相当 于 对 b 取 反 ， 然 后 将 两 数 相 加 */ 
13 public static int minus(int a, int b) { 


14 return a + negate(b); 

15 } 

要 对 数值 [的 取 反 ， 只 需 将 -1 连 加 砍 。 
2. 乘法 


加 法 和 乘法 之 间 关 系 也 同样 一 目 了 然 ，a 乘 以 5 其 实 就 是 将 a 连 加 5 次 。 
1 /* 将 a 连 加 b 次 ， 实 现 a 乘 b */ 


2 public static int multiply(int a, int b) { 
3 if (a < b) { 


图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 


9.7 数学 与 概率 183 





. return multiply(b，a); // 若 b < a， 算 法 会 比较 快 
5 } 

6 int sum = 0) 

7 for (int i = abs(b); i > 6; 1i--) { 
8 Sum += a; 

9 

16 if (b < 6@) { 

11 sum = negate(sum); 

12 } 

13 return sum; 

14 } 

15 


16 /* 返回 绝对 值 */ 
17 public static int abs(int a) { 
18 if (a < 6) { 


19 return negate(a); 
26 } else { 

21 return a; 

22 

23 } 


在 上 面 的 代码 中 ,有 个 地 方 必须 妥善 处 理 ， 就 是 负数 的 乘法 。 若 2 为 负数 ， 则 需 将 sum 的 值 正 
负 反 一 下 。 因 此 ， 这 段 代 码 实 际 上 是 这 么 回 事 : 

multiply(a, b) <-- abs(b) *+a* (-1 if b < 6). 

我 们 还 实现 了 一 个 简单 的 abs 辅 助 函 数 。 


3. 除法 

在 减 、 乘 、 除 三 种 运算 中 , 除法 无 疑 是 最 难 的 ,好 在 我 们 可 以 利用 已 有 的 multiply、 subtract 
和 negate 等 方法 实现 divide。 

除法 要 做 的 是 计算 x =a /5b 中 的 x。 或 者 ， 换 个 角度 来 说 ， 找 到 x， 使 得 a = bx。 这 人 么 一 来 ， 经 
过 变换 ， 这 个 问题 就 可 以 用 之 前 已 实现 的 乘法 运算 实现 。 

我 们 可 以 将 2 不 断 乘 以 逐 级 变 大 的 值 , 直至 得 到 a。 这么 做 非常 低 效 , 特别 是 前 面 的 multiply 
实现 含有 大 量 加 法 运算 。 

或 者 ， 我 们 可 以 好 好 利用 等 式 a = xb， 将 5 与 它 自身 连 加 直至 得 到 a， 就 能 算出 x。5 与 自身 连 
加 的 次 数 就 等 于 x 的 值 。 

当然 ,a 不 一 定 能 被 整除 ,这 也 没关系 。 这 个 问题 要 求实 现 的 是 整数 除法 ,本 来 就 应 该 对 结 
果 回 下 舍 人 (floor )。 

下 面 是 这 个 算法 的 实现 代码 。 












































int absa = abs(a); 
int absb = abs(b); 


1 public int divide(int a, int b) 

2 throws java.lang.ArithmeticException { 

3 if (b == 6) { 

4 throw new java.lang.ArithmeticException(“ERROR”); 
5 } 

6 

4 

8 
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9 int product = 6 
16 int x = 0; 
Ti while (product + absb <= absa) { /* 不 要 超过 a */ 


12 product += absb; 

13 X++; 

14 } 

15 

16 if ((a<68&8&b<6) ||(a>6868&b>6))1{ 
17 return x; 

18 } else { 

19 return negate(x); 

20 

21 } 


解决 此 题 时 ， 应 当 注 意 以 下 几 点 。 

口 多 想 想 乘法 和 除法 的 本 质 ， 以 逻辑 思考 的 方式 解 题 ， 这 么 做 很 管用 。 记 住 ， 所 有 (好 的 ) 

面试 题 都 能 以 逻辑 、 系 统 的 方式 解 出 来 ! 

口 面试 官 想 找 的 就 是 这 种 能 以 逻辑 思考 逐步 解决 问题 的 人 。 

口 这 是 个 让 你 展示 自己 能 够 写 出 干净 代码 的 绝 佳 问题 ， 特 别 是 显示 出 你 复 用 代码 的 能 力 。 
例如 ， 如 果 写 代码 时 没有 把 negate 独 立 出 来 ， 但 要 是 发 现 相关 代码 使 用 多 次 ， 就 应 该 将 
这 些 代码 写成 一 个 方法 。 

口 写 代码 时 要 小 心 假设 。 不 要 假设 所 有 数 都 是 正 数 ， 也 不 该 假设 a 会 比 5 大 。 


7.5 在 二 维 平面 上 ， 有 两 个 正方 形 ， 请 找 出 一 条 直线 ， 能 够 将 这 两 个 正方 形 对 半分 。 假 定 
正方 形 的 上 下 两 条 边 与 x 轴 平 行 。 (第 63 页 ) 

解法 

在 着 手 解 题 之 前 ， 有 必要 思考 一 下 题 中 一 条 “ 线 ” 的 准确 含义 。 一 条 线 是 由 斜率 和 y 轴 截 距 
确定 ?还 是 由 这 条 线 上 的 任意 两 点 定义 ? 抑或 , 所 谓 的 线 其 实 是 线段 , 以 正方 形 的 边 作 为 起 点 和 
终点 ? 

其 中 第 三 种 情况 会 让 此 题 变 得 更 有 意思 一 些 , 因此 这 里 假设 : 这 条 线 的 端点 应 该 落 在 正方 形 
的 边 上 。 在 面试 中 ， 你 应 该 与 面试 官 讨论 假设 条 件 。 


要 将 两 个 正方 形 对 半分 , 这 条 线 必须 连接 两 个 正方 形 的 中 心 点 。 利 用 slope = 2 一 力 ) 就 能 算 


XX 
出 和 斜率， 以 两 个 中 心 点 算出 斜率 后 ， 就 能 以 同一 公式 求 得 线段 的 起 点 和 终点 。 
在 下 面 的 代码 中 ， 假 设 原点 (0, 0) 位 于 左上 角 。 



























































public class Square { 


1 
2 oe 
3 public Point middle() { 

4 return new Point((this.left + this.right) / 2.6， 
5 (this.top + this.bottom) / 2.0); 
6 } 

7 

8 

9 


/* 返回 连接 mid1 和 mid2 的 线段 与 square 1 
* 的 边 相 交 的 点 ， 也 就 是 说 ， 从 mid2 到 mid1 
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一 条 线 ， 一 直 延 伸 直 至 碰 到 square 1 的 


傍 鲜 


public Point extend(Point mid1，Point mid2, double size) { 
/* 找 出 线段 mid2 -> mid1 的 方向 */ 
double xdir = midil.x < mid2.x ? -1 : 1; 
double ydir = midil.y < mid2.y ? -1 : 1; 


/* 若 mid1 和 mid2 的 x 坐标 相同 ， 计 算 入 率 时 
* 会 抛 出 除 替 异常， 因此 这 里 要 做 特别 的 处 理 
3 
if (mid1.x == mid2.x) { 
return new Pojint(mid1.x，mid1.y + ydir * size / 2.0); 


} 


double slope = (mid1.y - mid2.y) / (mid1.x - mid2.x); 
double x1 = 9; 
double y1 = 0@; 


/* 利用 算式 (yl - y2) / (x1 - x2) 计 算 针 率 (slope) 。 
* 注意 ， 若 斜率 很 "陡峭 ”(>1) ， 那 么 线段 的 终点 将 会 
* 碰 到 y 轴 上 距离 中 心 点 size / 2 的 位 置 。 若 斜率 
* 不 陡峭 (<1) ,那么 线段 的 终点 将 碰 到 X 轴 上 距 
* 离 中 心 点 size / 2 的 位 置 
*/ 

if (Math.abs(slope) == 1) { 

x1 = mid1.x + xdir * size / 2.08; 

y1 = mid1.y + ydir * size / 2.08; 
} else if (Math.abs(slope) < 1) { 

x1 = mid1.x + xdir * size / 2.08; 

y1 = slope * (x1 - mid1.x) + mid1.y; 
} else { 

y1 = mid1.y + ydir * size / 2.0; 

x1 = (yl - mid1.y) / slope + mid1.x; 
} 


return new Point(x1, y1); 


} 


public Line cut(Square other) { 
/* 计算 两 个 中 心 点 之 间 的 线段 与 正方 形 的 边 相 交 的 位 置 */ 
Point point 1 = extend(this.middle(), other.middle(), this.size); 
Point point 2 = extend(this.middle(), other.middle(), -1 * this.size); 
Point point 3 = extend(other.middle(), this.middle(), other.size); 


Point point 4 = extend(other.middle(), this.middle(), -1 * other.size); 


/* 在 上 面 这 些 点 中 ， 找 出 线段 的 起 点 和 终点 。 起 点 以 最 左边 且 在 上 方 的 为 准 ， 
* 终点 以 最 右边 且 在 下 方 的 为 准 */ 

Point start = point 1; 

Point end = point _ 1; 

Point[] points = {point 2, point 3, point 4}; 

for (int i = 6; i < points.length; i++) { 


start = points[i]; 


if (points[i].x < start.x || (points[i].x == start.x && points[i].y < start.y)) { 
} else if (points[i].x > end.x || (points[i].x == end.x && points[i].y > end.y)) { 9 
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64 end = points[i]; 
65 } 
66 } 


68 return new Line(start, end); 
69 } 
70 } 


此 题 意 在 考察 你 写 代 码 有 多 细心 ,毕竟 写 代码 时 很 容易 漏 掉 一 些 特殊 情况 ， 比 如 两 个 正方 形 
的 中 心 点 重合 。 着 手 解 题 之 前 ， 我 们 就 应 该 列 出 这 些 特殊 情况 ， 并 确保 予以 受 善 处 理 。 解 题 时 ， 
测试 必须 仔细 、 全 面 。 

7.6 在 二 维 平 面 上 ， 有 一 些 点 ， 请 找 出 经 过 点 数 最 多 的 那 条 线 。( 第 63 页 ) 

解法 

此 题 乍 一 看 很 简单 ， 老 实说 ， 确 实 有 点 。 

我 们 只 需 在 任意 两 点 之 间 “ 画 ”一 条 无 限 长 的 直线 ( 也 即 不 是 线段 )， 并 利用 散 列 表 追 踪 哪 
条 直线 出 现 次 数 最 多 。 这 种 做 法 的 时 间 复 杂 度 为 OOV] ， 因 为 一 共有 六 条 线段 。 

我 们 将 用 斜率 和 7 轴 截 距 而 不 是 两 个 点 来 表示 一 条 线 ， 这 样 一 来 ,检查 Cc, 加、Cc2, 四) 确定 的 
直线 是 否 等 于 (x;， y3) 到 |](xa, y4) 的 直线 就 相对 简单 。 

要 找到 出 现 次 数 最 多 的 直线 , 只 需 迭 代 遍 历 所 有 线段 ,并 用 散 列 表 数 出 每 条 直线 出 现 的 次 数 。 
够 简单 吧 !1 

不 过 ， 其 中 有 个 地 方 比较 棘手 。 首 先 ， 我 们 定义 ， 大 两 条 直线 的 和 斜率 和 ) 轴 截 距 相 同 ， 则 这 
两 条 直线 相等 。 接 着 ,我 们 会 基于 这 些 值 (确切 地 说 ,是 基于 和 斜率 ) 对 直线 进行 散 列 。 问 题 是 浮 
点 数 不 一 定 能 用 二 进 制 精 确 表 示 。 对 此 , 我 们 的 解决 办 法 是 检查 两 个 浮 点 数 的 差 值 是 否 在 某 个 极 
小 值 (epsilon ) 内 。 

对 散 列表 而 言 , 这 又 意味 着 什么 呢 ? 这 意味 着 , 斜率 “相等 ”的 两 条 直线 , 散 列 值 未 必 相 同 。 
为 此 ， 我 们 将 把 斜率 减 去 一 个 极 小 值 ， 并 以 得 到 的 结果 flooredSlope 作 为 散 列 键 。 然 后 ， 要 取 
得 所 有 可 能 相等 的 直线 ， 我 们 会 搜索 三 个 位 置 : flooredslope 、flooredslope - epsilon 和 
flooredSlope + epsilon。 这 能 确保 我 们 已 检查 了 所 有 可 能 相等 的 直线 。 














































































































1 Line findBestLine(GraphPoint[] points) { 

p2 Line bestLine = null; 

3 int bestCount = 0@; 

4 HashMap<Double, ArrayList<Line>> linesBySlope = 
5 new HashMap<Double, ArrayList<Line>>(); 
6 
2 
8 


for (int i = 6; i < points.length; i++) { 
for (int j = i + 1; j < points.length; j++) { 


9 Line line = new Line(points[i], points[j]); 

16 insertLine(linesBySlope, line); 

二 int count = countEquivalentLines(linesBySlope, line); 
12 if (count > bestCount) { 

13 bestLine = line; 

14 bestCount = count; 

15 } 
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} 


} 
} 


return bestLine; 


int countEquivalentLines(ArrayList<Line> lines, Line line) { 


} 


if (lines == null) return 6; 
int count = 0@; 
for (Line parallelLine : lines) { 
if (parallelLine.isEquivalent(line) count++; 


} 


return count; 


int countEquivLines(HashMap<Double, ArrayList<Line>> linesBySlope, Line line) { 


} 


double key = Line.floorToNearestEpsilon(line.slope); 

double eps = Line.epsilon; 

int count = countEquivalentLines(linesBySlope.get(key), line) + 
countEquivalentLines(linesBySlope.get(key - eps), line) + 
countEquivalentLines(linesBySlope.get(key + eps), line); 

return count; 


void insertLine(HashMap<Double，ArrayList<Line>> linesBySlope, 


} 


Line line) { 
ArrayList<Line> lines = null; 
double key = Line.floorToNearestEpsilon(line.slope); 
if (!linesBySlope.containsKey(key)) { 
lines = new ArrayList<Line>(); 
linesBySlope.put(key, lines); 
} else { 
lines = linesBySlope.get(key); 
} 


lines.add(line); 


public class Line { 


public static double epsilon = .88601; 
public double slope, intercept; 
private boolean infinite slope = false; 


public Line(GraphPoint p, GraphPoint q) { 
if (Math.abs(p.x - q.x) > epsilon) { // 若 两 个 点 的 x 坐标 不 同 
slope = (p.y - q.y) / (p.x - 9q.X); // 计算 儿 率 
intercept = p.y - Slope * p.x; // 利用 y=mx+b 计 算 y 轴 截 距 
} else { 
infinite slope = true; 
intercept = p.x; // x 轴 截 距 ， 因 为 斜率 无 穷 大 
} 
} 


public static double floorToNearestEpsilon(double d) { 
int r = (int) (d / epsilon); 
return ((double) r) * epsilon; 


图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 








70 } 

71 

72 public boolean isEquivalent(double a, double b) { 
73 return (Math.abs(a - b) < epsilon); 

74 } 

75 

76 public boolean isEquivalent(Object o) { 

77 Line 1 = (Line) o; 

78 if (isEquivalent(1.slope, slope) && 

79 isEquivalent(1.intercept, intercept) && 
80 (infinite_slope == 1.infinite slope)) { 
81 return true; 

82 } 

83 return false; 

84 } 

85 } 





计算 直线 的 斜率 务必 小 心间 慎 。 直 线 有 可 能 完全 垂直 ， 也 即 它 没有 y 轴 截 距 且 和 斜率 无 穷 大 。 
我 们 可 以 用 单独 的 标记 ( infinite_slope ) 跟踪 记录 。 在 equals 方 法 中 ， 必 须 检查 这 个 条 件 。 


7.7 ”有 些 数 的 素 因 子 只 有 3、5、7， 请 设计 一 个 算法 ， 找 出 其 中 第 k 个 数 。( 第 63 页 ) 


解法 
根据 定义 ， 这 些 数字 看 起 来 都 像 是 3 * 5” * 7°。 
下 面 先 列 出 符合 该 形式 的 数字 ， 此 题 要 求 找 出 这 种 数字 里 的 第 k 个 。 











1 2 38 * 598 * 78 
3 3 3 
5 5 5 
7 7 38 * 58 * 71L 
9 3*3 3 
15 3*5 3 * 5 * 7 
21 3*7 3 * 59 * 7 
25 5*5 3 5 7 
1 总 38 * 58 * 78 
27 3*9 32 * 二 
35 5*7 3 
45 5*9 3* YD 7 
49 7XV 人 
63 3*21 37 信和 


出 于 3 5 和 国 此 3 5 * 7 粹 定 已 下 我 们 的 列表 申 出 现 过 。 实际 上 下 
面 这 些 值 已 在 列表 中 出 现 过 了 : 
口 3 *# S2*# 72 
口 3"* 5 和 # 72 
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口 3 * 5? * 7c-1 
男 一 种 思路 是 ， 所 有 数字 都 可 以 表示 成 如 下 形式 : 
口 3* (列表 中 之 前 出 现 的 某 个 数 ) 
口 5* (列表 中 之 前 出 现 的 某 个 数 ) 
口 7* (列表 中 之 前 出 现 的 某 个 数 ) 

由 此 可 知 ，Ax 可 以 表示 为 (3、5 或 7) * ({Ai, …, Ac 中 的 某 个 值 )。 另 外 ， 根 据 定义 可 知 ，Ax 
是 列表 中 的 下 一 个 数 。 因 此 ，A 将 是 最 小 的 新 增 数字 ( 其 余 更 小 的 数 已 在 {A …,Ae 二 中 )， 可 以 
通过 将 列表 中 的 每 个 值 与 3 、5 或 7 相 乘 得 到 。 

怎样 才能 找到 Ai 实际 上 ， 我 们 可 以 将 列表 中 的 数字 与 3、5 和 7 相 乘 ， 找 出 还 未 加 入 列表 的 
最 小 数 。 这 种 解法 的 时 间 复 杂 度 为 OC)。 不 算 太 糟 ， 不 过 我 想 还 可 以 做 得 更 好 。 

之 前 我 们 曾 试 着 从 列表 中 的 元 素 “ 拉 出 ”Ax( 将 这 些 元 素 与 3、5 和 7 相 乘 ), 其 实 可 以 换个 思路 ， 
可 以 让 列表 中 的 元 素 “推出 ”三 个 后 续 值 ， 也 就 是 说 ， 列 表 中 的 每 个 数 A 终 将 以 下 列 形式 出 现 : 
D3*A, 
DS*A, 
D7*A; 

照 着 这 个 思路 事先 做 好 准备 ,每 次 要 将 A 加 入 列表 时 ， 就 用 某 个 临时 列表 存放 3A、5A; 和 7A， 
三 个 值 。 要 产生 Ar 时， 我 们 会 搜索 这 个 临时 列表 ， 找 出 最 小 的 值 。 



































我 们 的 代码 大 致 如 下 : 

1 public static int removeMin(Queue<Integer> q) { 
2 int min = q.peek(); 

3 for (Integer v : q) { 

4 if (min > v) { 

5 min = v; 

6 } 

py 

8 while (q.contains(min)) { 
9 q.remove(min); 

16 } 

11 return min; 

12 } 

13 


14 public static void addProducts(Queue<Integer> q, int v) { 
5 q.add(v * 3); 

16 q.add(v * 5); 

17 q.add(v * 7); 

18 } 


26 public static int getkthMagicNumber(int k) { 
21 if (k < 6) return ©; 


23 int val = 1; 

24 Queue<Integer> q = new LinkedList<Integer>(); 
25 addProducts(q, 1); 

26 for (int i = 8; i < ki i++) { 
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和 27 val = removeMin(q); 
28 addProducts(q, val); 
29 } 

36 return val; 

31 } 


相 比 第 一 种 解法 ， 这 个 算法 确实 要 好 得 多 ， 但 仍 不 够 完美 。 
为 了 产生 新 元 素 A;， 我 们 会 搜索 一 整个 链表 ， 其 中 每 个 元 素 类 似 如 下 形式 之 一 : 
口 3* 之 前 的 元 素 
口 5* 之 前 的 元 素 
口 7* 之 前 的 元 素 
我 们 还 可 以 优化 掉 哪些 无 谓 的 操作 ? 
假设 有 如 下 列表 。 
qe = {7A1, 5Az, 7A2, 7A3, 3As, 5As, 7As, 5As, 7As} 
要 在 列表 中 查找 最 小 值 时 ， 先 检查 7A1 < min 是 否 成 立 ， 然 后 检查 7A; < min。 这 看 起 来 有 点 
笨拙 ， 是 不 是 ”既然 已 经 知道 A1 <A;， 因 此 只 需 检 查 7A1 即 可 。 
若 从 一 开始 就 按 常 数 因子 将 列表 分 组 存放 ， 那 就 只 需 检 查 3、5 和 7 倍数 的 第 一 个 ， 后 续 元 素 
一 定 比 第 一 个 元 素 大 。 
也 就 是 说 ， 上 面 的 列表 应 该 如 下 : 
Q36 = {3A4} 
Q56 = {5A,, 5As, 5As} 
Q76 = {7A1, 7As, 7A3, 7As, 7As} 
要 求 得 最 小 值 ， 我 们 只 需 检 查 各 个 队列 的 队 首 元 素 。 
y= min(Q3.head(), Q5.head0, Q7.head()) 
求 出 y 后 ， 就 要 把 3y 插 入 Q3、5y 插 入 Q5、7y 插 入 Q7。 不 过 ， 只 有 这 些 元 素 在 其 他 列表 中 不 存 
在 时 ,我们 才 会 将 它们 插入 列表 。 
举 个 例子 ,为 什么 3y 可 能 已 经 存在 某 个 队列 中 ? 很 简单 ， 如 果 y 是 从 Q7 拉 出 来 的 ， 就 表示 y= 
7x，x 是 某 个 较 小 的 值 。 如 果 7x 是 最 小 值 ， 那 么 ， 我 们 一 定 碰 到 过 3x。 磁 到 3x 时 会 怎么 做 呢 ? 我 
们 会 将 7* 3x 插 入 Q7。 注 意 ，7* 3x = 3 * 7x = 3y。 
换 名 话说， 如果 从 Q7 拉 出 一 个 元 素 ， 它 看 起 来 像 7* suffix， 而 我 们 知道 已 处 理 过 3 * suffix 和 
5* suffix。 处 理 3* suffix 时 ， 将 7* 3* suffix 插 入 Q7。 而 处 理 5 * suffix 时 ， 我 们 知道 已 经 将 7 * 5 * 
suffix 插 入 Q7。 至 此 ， 唯 一 还 未 碰 到 的 值 是 7 * 7 * suffix， 因 此 我 们 只 会 将 7 * 7 * suffix 插 入 Q7。 
下 面 我 们 会 举例 说 明 ， 真 正 做 到 心 知 肚 明 。 












































一 开始 : 
Q3 = 3 
Q5 = 5 
Q7 = 7 
取出 min = 3，Q3 插 入 3*3，Q5 揪 入 5+3，Q7 插 入 7#3。 
Q3 = 3*3 
Q5 = 5, 5*3 
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Q7 = 7, 7*3 
取出 min = 5，3*#5 重 复 了 ， 因 为 我 们 已 经 处 理 过 5*3。Q5 村 入 5+k5，Q7 持 入 7#+5。 
Q3 = 3*3 
Q5 = 5*3, 5*5 
Q7 = 7, 7*3, 7*5. 
取出 min = 7，3*7 和 5*7 重 复 了 ， 因 为 已 处 理 过 7*3 和 7*5。Q7 插 入 7*7。 
Q3 = 3*3 
Q5 = 5*3, 5*5 
Q7 = 7*3, 7*5, 7*7 


取出 min = 3*3 = 9，Q3 桂 入 3*+3*3，Q5 插 入 3*3*5，Q7 持 入 3*3*7。 


Q3 = 3*3*3 
Q5 = 5+3, 5*5, 5*3*3 
Q7 = 7+3, 7*5, 7*7, 7*3*3 


取出 min = 5*3 = 15，3*(5*3) 重 复 了 ， 因 为 已 处 理 过 5*(3*3)。Q5 桂 入 5*5*3，Q7 桂 入 7*5*3。 


Q3 = 3+*3*3 
Q5 = 5*5, 5*3*3, 5*5*3 
Q7 = 7+3, 7*5, 7T*7, 7*3*3, 7*5*3 


取出 min = 7*3 = 21，3*(7*3) 和 5*(7*3) 重 复 了 ， 因 为 已 处 理 过 7*(3*3) 和 7*(5*3)。Q7 插 入 7*7*3。 


Q3 = 3*3*3 

Q5 = 5*5, 5S*3*3, 5*5*3 

Q7 = 7*5, 7+7, 7*3*3, 7*5*3, 7*7*3 
此 题解 法 的 伪 码 如 下 。 
(1) 初始 化 array 和 队列 : Q3 、Q5 和 Q7。 
(2) 将 1 插入 array。 





(3) 分 别 将 1*3、1*5 和 1*7 插 入 Q3 、Q5 和 Q7。 
(4) 令 x 为 Q3、Q5 和 Q7 中 的 最 小 值 。 将 x 添加 至 array 尾 部 。 
(5) 若 x 存 在 于 : 
 Q3， 则 将 x*3、x*5 和 x*7 放 入 Q3、Q5 和 Q7， 从 Q3 移 除 x。 
 Q5， 则 将 x*5 和 x*7 放 入 Q5 和 Q7， 从 Q5 移 除 x。 
 Q7， 则 只 将 x*7 放 入 Q7， 从 Q7 移 除 x。 
(6) 重复 步骤 4~6， 直 至 找到 第 k 个 元 素 。 
下 面 是 该 算法 的 实现 代码 。 























1 public static int getkthMagicNumber(int k) { 

2 if (k < 6) 1{ 

3 return 9) 

4 } 

5 int val = ©; 

6 Queue<Integer> queue3 = new LinkedList<Integer>(); 
7 Queue<Integer> queue5 = new LinkedList<Integer>(); 
8 Queue<Integer> queue7 = new LinkedList<Integer>(); 
9 queue3.add(1); 

16 


11 /* 从 6 到 K 的 选 代 */ 
12 for (int i = 6;j i <= k; i++) { 


13 int v3 = queue3.size() > 6 ? queue3.peek() : 
14 Integer .MAX_VALUE:; 
15 int v5 = queue5.size() > 6 ? queue5.peek() : 
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16 Integer .MAX_VALUE; 
17 int v7 = queue7.size() > 6 ? queue7.peek() : 
18 Integer .MAX_VALUE; 
19 val = Math.min(v3, Math.min(v5, Vv7)); 
26 if (val == V3) { // 放 入 队列 3、 队 列 5 和 队列 7 
2 queue3 .remove() ; 
22 queue3.add(3 * val); 
23 queue5.add(5 * val); 
24 } else if (val == v5) { // 放 入 队列 5 和 队列 7 
25 queue5 .remove(); 
26 queue5.add(5 * val); 
27 } else if (val == V7) { // 放 入 队列 7 
28 queue7 .remove(); 
29 } 
36 queue7.add(7 * val); // 总 是 放 入 队列 7 
31 } 
32 return val; 
33 } 





碰 到 这 个 问题 时 ， 尽 最 大 努力 去 解决 ， 虽 然 问题 确实 有 难度 。 你 可 以 先 从 蛮 力 法 开始 (有 挑 


战 性 ， 但 不 那么 辐 手 )， 然 后 试 着 不 断 优 化 。 或 者 ， 试 着 从 这 些 数 中 找 出 规律 。 








当 你 解 题 卡 壳 时 ， 面 试 官 有 可 能 会 帮 你 一 把 。 不 管 怎样 ， 绝 不 要 放弃 ! 大 声 说 出 你 的 思路 、 





疑问 ， 并 解释 你 的 思考 过 程 。 面 试 官 





或 许 就 会 介入 指导 一 下 。 





记 住 ,面试 官 并 不 期 待 你 给 出 完美 无 缺 的 解法 ,而 是 会 对 照 其 他 求职 者 来 评估 你 的 表现 。 面 








对 刁钻 的 问题 ， 大 家 都 需 拼 尺 全 力 。 


9.8 面向 对 象 设计 


8.1 请 设计 用 于 通用 扑克 牌 的 数据 结构 。 并 说 明 你 会 如 何 创建 该 数据 结构 的 子 类 ， 实 现 


“二 十 一 点 ”游戏 。( 第 66 页 ) 
解法 


首先 ， 看 得 出 来 所 谓 的 “通用 ”扑克 牌 隐 含 有 不 少 信息 。 这 里 的 “通用 ”可 以 指 能 


殉 牌 游戏 的 标准 扑克 牌 组 ， 也 可 以 扩展 为 Uno 牌 或 棒球 卡 。 面 试 时 记得 询问 面试 


体 含义 ,这 点 很 重要 。 























官 “ 请 


通用 ”的 具 





卢 ! 























用 来 玩 扑 





假设 面试 官 说 清楚 了 ,这 是 一 副 标 准 纸牌 ， 一 共 52 张 ， 就 如 同 你 在 二 十 一 点 或 扑克 牌 游戏 中 


使 用 的 牌 组 。 这 样 一 来 ， 整 个 设计 大 


public enum Suit { 


致 如 下 : 


Club (686)，Diamond (1), Heart (2),Spade (3); 


private int value; 


public int getValue() { return value; } 


public static Suit getSuitFromValue(int value) { ... } 


} 


1 
2 
3 
4 private Suit(int v) { value = v; } 
5 
6 
7 
8 
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9 public class Deck <T extends Card> { 


16 
过 
12 
13 
14 
15 
16 
和 
18 
19 
26 
21 } 
22 


private ArrayList<T> cards; // 所 有 有 牌 ， 包 括 已 经 发 出 去 的 ， 还 未 发 出 去 的 
private int dealtIndex = 6; // 标示 第 一 张 还 未 发 出 去 的 牌 


public void setDeckofCards(ArrayList<T> deckofCards) { ... } 
public void shuffle() { ... } 


public int remainingCards() { 
return cards.size() - dealtIndex; 


} 
public T[] dealHand(int number) { ... } 
public T dealCard() { ... } 


23 public abstract class Card { 


24 
25 
26 
27 
28 
29 
36 
31 
32 
33 
34 
35 
36 
37 
38 
39 
40 
41 
42 
43 
44 
45 } 
46 


private boolean available = true; 


/* 牌 面 的 数字 或 人 头 ， 数 字 2 到 18，11 为 杰克 ， 
# 12 为 皇后 ，13 位 国王 ，1 为 Ace */ 
protected int faceValue; 

protected Suit suit; 


public Card(int c, Suit s) { 
faceValue = c; 
suit = s; 

} 

public abstract int value(); 


public Suit suit() { return suit; } 


/* 检查 这 张 牌 是 否 发 给 某 个 人 */ 
public boolean isAvailable() { return available; } 
public void markUnavailable() { available = false; } 


public void markAvailable() { available = true; } 


47 public class Hand <T extends Card> { 


48 
49 
50 
51 
52 
53 
54 
55 
56 
57 
58 
59 
66 
61 } 


protected ArrayList<T> cards = new ArrayList<T>(); 


public int score() { 
int score = 0@; 
for (T card : cards) { 
score += card.value(); 


} 


return score; 


} 


public void addCard(T card) { 
cards.add(card); 


} 
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在 上 面 的 代码 中 , 我 们 以 泛 型 实现 了 peck, 同时 把 T 的 类 型 限定 为 Card。 另外 , 我 们 还 将 Card 
实现 成 抽象 类 , 这 是 因为 如 果 不 知 道 玩 的 是 什么 游戏 , 诸如 value() 的 方法 就 没有 太 大 意义 。( 你 
可 能 会 据 理 力争 ， 认 为 这 些 方 法 还 是 应 该 实现 为 好 ， 以 标准 标准 扑克 牌 规则 实现 默认 值 。) 

现在 , 假设 要 构建 二 十 一 点 游戏 , 我 们 需要 知道 这 些 牌 的 数值 。 人头 牌 区 、Q 、J 等 于 10，Ace 
为 11 (大 部 分 情况 下 为 11 ， 不 过 这 应 该 交 由 Hand 类 负责 ， 而 不 是 交 给 下 面 这 个 类 )。 





















































1 public class Black]JackHand extends Hand<BlackJackCard> { 
2 /” 在 二 十 一 点 玩法 中 ， 一 手 牌 可 以 有 多 种 分 数 ， 因 为 

3 * Ace 具 有 多 个 数值 。 若 低 于 21 就 返回 最 高 的 分 数 ， 

4 * 若 高 过 21 就 返回 最 低 的 分 数 */ 

5 public int score() { 

6 ArrayList<Integer> scores = possibleSscores(); 

了 int maxUnder = Integer.MIN VALUE; 

8 int minOver = Integer.MAX_VALUE; 

9 for (int score : scores) { 

16 if (score > 21 && score < minOver) { 

汪 玫 minOver = score; 

12 } else if (score <= 21 && score > maxUnder) { 

13 maxUnder = score; 

14 } 

15 } 

16 return maxUnder == Integer.MIN VALUE ? minOver : maxUnder; 
和 7 } 

18 


19 /* 返回 一 个 列表 ， 包 含 这 手 牌 所 有 可 能 的 分 数 
20 * (将 Ace 当 作 1 和 11 进 行 计算 ) */ 
21 private ArrayList<Integer> possibleScores() { ... } 


23 public boolean busted() { return score() > 21; } 
24 public boolean is21() { return score() == 21; } 
25 public boolean isBlackJack() { ... } 

26 } 


28 public class BlackJackCard extends Card { 
29 public BlackJackCard(int c, Suit s) { super(c, s); } 
36 public int value() { 


3 if (isAce()) return 1; 
32 else if (faceValue >= 11 && faceValue 《= 13) return 16; 
33 else return faceValue; 
34 } 

35 

36 public int minValue() { 
37 if (isAce()) return 1; 
38 else return value(); 

39 } 

46 

41 public int maxValue() { 
42 if (isAce()) return 11; 
43 else return value(); 

44 } 

45 


46 public boolean isAce() { 
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47 return faceValue == 1; 
48 } 


56 public boolean isFaceCard() { 
51 return faceValue >= 11 && faceValue <= 13; 
52 } 

















这 只 是 Ace 的 一 种 处 理 方式 ， 另 一 种 做 法 是 创建 一 个 继承 自 BlackJackCard 的 Ace 类 。 
在 本 书 所 附 、 可 下 载 的 代码 中 ， 提 供 了 一 个 可 自动 执行 的 二 十 一 点 游戏 程序 。 


8.2 ”设想 你 有 个 呼叫 中 心 ， 员 工分 成 三 个 层级 : 接线 员 、 主 管 和 经 理 。 客 户 来 电 会 先 分 配 
给 有 空 的 接线 员 。 若 接线 员 处 理 不 了 ， 就 必须 将 来 电 往 上 转 给 主管 。 若 主管 没 空 或 是 无 法 处 理 ， 
则 将 来 电 往 上 转 给 经 理 。 请 设计 这 个 问题 的 类 和 数据 结构 ， 并 实现 一 个 dispatchcal1() 方 法 ， 
将 客户 来 电 分 配给 第 一 个 有 空 的 员工 。( 第 66 页 ) 


解法 

三 个 员工 层级 各 有 各 的 职责 ， 因此 , 不 同 层级 会 有 专门 的 函数 。 我 们 应 该 将 它们 放 在 各 自 对 
应 的 类 里 。 

有 些 东 西 是 所 有 员工 都 有 的 ,比如 地 址 、 姓 名 、 职 位 和 年 龄 等 。 这 些 东西 可 以 放 在 一 个 类 里 ， 
再 由 其 他 类 扩展 或 继承 。 

最 后 ， 还 应 该 有 一 个 CallHandler 类 ， 负 责 将 来 电 分 派 给 合适 的 负责 人 。 

注意 , 任何 面向 对 象 设 计 问 题 ， 都 会 有 很 多 不 同 的 对 象 设 计 方 式 。 请 跟 面 试 官 讨 论 各 种 设计 
方案 的 优 劣 。 通 常 ， 设 计时 应 该 从 长 远 考 虑 ， 注 重 代 码 的 灵活 性 和 可 维护 性 。 

下 面 我 们 将 详细 说 明 每 个 类 。 

CallHandler 实 现 为 一 个 单 态 类 ， 它 是 程序 的 主体 ， 所 有 来 电 都 先 由 这 个 类 进行 分 派 。 





























1 public class CallHandler { 

2 private static CallHandler instance; 

3 

4 /* 三 个 员工 层级 : 接线 员 、 主 管 、 经 理 */ 

5 private final int LEVELS = 3; 

6 

2 /* 起 始 设 定 16 位 接线 员 、4 位 主管 和 2 位 经 理 */ 
8 private final int NUM_RESPONDENTS = 16; 
9 private final int NUM_MANAGERS = 4; 

16 private final int NUM_DIRECTORS = 2; 

11 

12 ” /* 员工 列表 ， 以 层级 区 分 : 

13 * employeeLevels[6] = 接线 员 

14 * employeeLevels[1] = 主管 

15 * employeeLevels[2] = 经 理 

16 */ 

17 List<List<Employee>> employeeLevels; 

18 


19 /* 存放 来 电 层 级 的 队列 */ 
26 List<List<Call>> callQueues; 
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22 protected CallHandler() { ... } 


24 /* 取得 单 态 类 的 实例 */ 
25 public static CallHandler getInstance() { 


26 if (instance == null) instance = new CallHandler(); 
27 return instance; 

28 } 

29 

36 /* 找 出 第 一 个 有 空 处 理 来 电 的 员工 */ 

31 public Employee getHandlerForCall(Call call) { ... } 
32 


33 /* 将 来 电 分 派 给 有 空 的 员工 ， 若 没 人 有 空 ， 
34 * 就 存放 在 队列 中 */ 
35 public void dispatchCall(Caller caller) { 


36 Call call = new Call(caller); 
37 dispatchCall(call); 

38 } 

39 


46 /* 将 来 电 分 配给 有 空 的 员工 ， 若 没 人 有 空 ， 
41 * 就 存放 在 队列 中 */ 
42 public void dispatchCall(Call call) { 


43 /* 试 着 将 来 电 分派 给 层级 最 低 的 员工 */ 

44 Employee emp = getHandlerForCall(call); 

45 if (emp != null) { 

46 emp.receiveCall(call); 

47 call.setHandler (emp); 

48 } else { 

49 /* 根据 来 电 级 别 ， 将 来 电 放 到 相应 的 

56 * 队列 中 */ 

51 call.reply(“Please wait for free employee to reply”); 
52 callQueues[call.getRank().getValue()].add(call); 
53 } 

54 } 

55 

56 /* 有 员工 有 空 了 ， 查 找 该 员工 可 服务 的 来 电 。 

57 * 若 分 派 了 来 电 则 返回 true， 否 则 返回 false */ 

58 public boolean assignCall(Employee emp) { ... } 

59 } 











Call 代 表 客 户 来 电 , 每 次 来 电 会 有 个 最 低层 级 ,并 且 会 被 分 派 给 第 一 个 可 处 理 该 来 电 的 员工 。 


public class Call { 
/* 可 处 理 此 来 电 的 最 低层 级 员工 */ 
private Rank rank; 





/* 拨号 方 */ 
private Caller caller; 


/* 处 理 来 电 的 员工 */ 
private Employee handler; 


oONUUAWwWDOP 


1 public Call(Caller c) { 
12 rank = Rank.Responder; 
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13 caller = Cc; 
14 } 


16 /* 设 定 处 理 来 电 的 员工 */ 
17 public void setHandler(Employee e) { handler = e; } 


19 public void reply(String message) { ... } 
26 public Rank getRank() { return rank; } 
21 public void setRank(Rank r) { rank = Pi } 


22 public Rank incrementRank() { ... } 
23 public void disconnect() { ... } 
24 } 


Employee 是 Director、Manager 和 Respondent 类 的 父 类 , 由 于 没有 必要 直接 实例 化 Employee 
因此 是 个 抽象 类 。 








1 abstract class Employee { 

2 private Call currentCall = null; 

3 protected Rank rank; 

4 

5 public Employee() { } 

6 

7 /* 开始 交谈 对 话 */ 

8 public void receiveCall(Call call) { ... } 
9 

16 /* 问题 解决 了 ， 结 束 来 电 #/ 

11 public void callCompleted() { ... } 
12 


13 /* 问题 未 解决 ， 往 上 转 给 更 高 层级 的 员工 ， 
14 * 并 为 该 员工 分 派 新 的 来 电 */ 


15 public void escalateAndReassign() { ... } 
16 } 

17 

18 /* 分 派 新 的 来 电 给 该 员工 ， 若 他 有 空 的 话 */ 

19 public boolean assignNewCall() { ... } 

20 

21 /* 返回 该 员工 是 否 有 空 */ 

22 public boolean isFree() { return currentCall == null; } 
23 

24 public Rank getRank() { return rank; } 

25 } 

26 


有 了 Employee 类 ，Respondent、Director 和 Manager 只 是 在 此 基础 上 稍微 扩展 一 下 。 


class Director extends Employee { 
public Director() { 
rank = Rank.Director; 


} 


class Manager extends Employee { 
public Manager() { 


1 
2 
3 
4 
5 } 
6 
了 
8 
9 rank = Rank.Manager; 
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13 class Respondent extends Employee { 
14 public Respondent() { 
15 rank = Rank.Responder; 


上 面 只 是 此 题 的 一 种 设计 方式 。 注 意 ， 其 实 还 有 其 他 许多 同样 不 错 的 方法 。 
在 面试 中 , 要 写 这 么 多 代码 似乎 有 点 可 怕 ,， 确实 如 此 。 这 里 给 出 的 代码 比较 完整 ， 在 实际 面 
试 中 ， 可 能 不 需要 写 得 这 么 全 ， 有 些 细 方 可 以 先 简略 带 过 ， 等 到 有 时 间 了 再 作 补 充 。 


8.3 ”运用 面向 对 象 原则 ， 设 计 一 款 音 乐 点 唱机 。( 第 66 页 ) 


解法 
但 几 遇 到 面向 对 象 设计 的 问题 , 一 开始 就 要 向 面试 官 问 几 个 问题 , 以 便 厘清 设计 时 有 哪些 限 
制 条 件 。 这 台 点 唱机 放 的 是 CD 吗 ? 是 唱片 ? 还 是 MP3? 它 是 计算 机 模拟 软件 ， 还 是 代表 一 台 实 
体 点 唱机 ? 播放 音乐 要 收 钱 还 是 免费 ? 收 钱 的 话 ， 要 求 哪 国货 币 ” 可 以 找 零 吗 ? 
遗憾 的 是 ， 这 里 没有 面试 官 ， 我 们 无 法 与 之 对 话 。 因 此 ,下 面 将 作出 一 些 假设 。 假设 这 台 点 
唱机 为 计算 机 模拟 软件 ， 与 实体 点 唱机 非常 相像 ， 另 外 ， 假 定 播放 音乐 是 免费 的 。 
至 此 人 尘埃 落 定 ， 下 面 将 列 出 基本 的 系统 组 件 : 
口 点 唱机 (Jukebox ); 
口 CD ; 
口 歌曲 ( Song ); 
口 艺术 家 〈 Artist ); 
口 播放 列表 ( Playlist ); 
口 显示 屏 (Display， 在 屏幕 上 显示 详细 信息 )。 
接 下 来 ， 进 一 步 分 解 上 述 组 件 ， 考 虑 可 能 的 动作 。 
口 新 建 播放 列表 ( 包括 新 增 、 删 除 和 随机 播放 ) 
口 CD 选择 器 
口 歌曲 选择 器 
口 将 歌曲 放 进 播放 队列 
口 获取 播放 列表 中 的 下 一 首 歌曲 
另外 ， 还 可 引入 用 户 : 
口 添加 ; 
口 删除 ; 
口 信用 信息 。 
每 个 主要 系统 组 件 大 致 都 会 转换 成 一 个 对 象 , 而 每 个 动作 则 转换 为 一 个 方法 。 下 面 将 介绍 一 
种 可 行 的 设计 。 
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Jukebox 类 代表 此 题 的 主体 ， 系 统 各 个 组 件 之 间或 系统 与 用 户 间 的 大 量 交 互 ， 都 是 通过 这 个 
类 实现 的 。 


1 public class Jukebox { 

2 private CDPlayer cdPlayer; 

3 private User user; 

4 private Set<CD> cdCollection; 

5 private SongSelector ts; 

6 

7 public Jukebox(CDPlayer cdPlayer, User user, 
8 Set<CD> cdCollection, SongSelector ts) { 
9 Se 

16 } 

11 

12 public Song getCurrentSong() { 

13 return ts.getCurrentSong(); 

14 } 

15 

16 public void setUser(User u) { 

17 this.user = U; 

18 } 

19 } 


跟 实际 CD 播放 还 一 样 ，CDPlayer 类 一 次 只 能 放 一 张 CD。 不 在 播放 的 CD 都 存放 在 点 唱机 里 。 


public class CDPlayer { 
private Playlist p; 
private CD c; 


1 

它 

3 

4 

5 /* 构造 函数 */ 
6 public CDplayer(CD c, Playlist p) { ... } 

;4 public CDplayer(Playlist p) { this.p = p; } 

8 public CDplayer(CD c) { this.c = c; } 

9 

16 /* 播放 歌曲 */ 

11 public void playSong(Song s) {...} 

12 

13 /* getter 和 setter */ 

14 public Playlist getPlaylist() { return p; } 

15 public void setplaylist(Playlist p) { this.p = p; } 
16 

17 public CD getCD() { return c; } 

18 public void setCD(CD c) { this.c = c; } 

19 } 


Playlist 类 管理 当前 播放 的 歌曲 和 待 播放 的 下 一 首 歌曲 。 它 本 质 上 是 播放 队列 的 包 囊 类 , 还 
提供 了 一 些 操作 起 来 更 方便 的 方法 。 
public class Playlist { 
private Song song; 


1 
2 
3 private Queue<Song> queue; 

4 public Playlist(Song song, Queue<Song> queue) { 
5 i 

6 





} 
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7 public Song getNextSToplay() { 

8 return queue.peek(); 

9 } 

16 public void queueUpSong(Song s) { 
1 queue.add(s); 

12 } 

13 } 


CD、Song 和 User 这 几 个 类 都 相当 简单 ， 主 要 由 成 员 变 量 、getter ( 访问 ) 和 setter (设置 ) 方 
法 组 成 。 


public class CD { 
/* 识别 码 、 艺 术 家 、 歌 曲 等 */ 
} 


public class Song { 
/* 识别 码 、CD 〈 可 能 为 空 ) 、 名 称 、 长 度 等 */ 
} 


9 public class User { 

16 private String name; 

11 public String getName() { return name; } 

12 public void setName(String name) { this.name = name; } 
13 public long getID() { return ID; } 

14 public void setID(long iD) { ID = iD; } 

15 private long ID; 


16 public User(String name, long iD) { ... } 

17 public User getUser() { return this; } 

18 public static User addUser(String name, long iD) { ... } 
19 } 


这 当然 绝 非 唯一 “正确 ”的 实现 。 跟 其 他 限制 条 件 一 样 ， 面 试 官 对 一 开始 询问 的 回应 也 会 影 
响 点 唱机 里 各 种 类 的 设计 。 


8.4 ”运用 面向 对 象 原则 ， 设 计 一 个 停车 场 。( 第 66 页 ) 


解法 
这 个 问题 的 表述 有 些 含糊 ， 在 实际 的 面试 中 也 会 出 现 这 种 情况 。 这 就 要 求 你 与 面试 官 交 流 ， 
青 楚 允许 哪些 车 辆 进入 停车 场 ， 它 是 不 是 多 层 的 ， 等 等 
为 便于 描述 , 我 们 先 做 如 下 假设 条 件 。 这 些 特定 的 假设 条 件 会 让 问题 变 得 更 复杂 , 但 又 不 致 
过 于 复杂 。 如 果 你 想 作 出 其 他 假设 ， 那 也 完全 不 成 问题 。 
口 停车 场 是 多 层 的 。 每 一 层 有 好 几 排 停车 位 。 
口 停车 场 可 停放 摩托 车 、 轿 车 和 大 巴 。 
口 停车 场 有 摩托 车 车 位 、 小 车 位 和 大 车 位 。 
口 摩托 车 可 停 在 任意 车 位 上 。 
口 轿车 可 停 在 单个 小 车 位 或 大 车 位 上 。 
口 大 巴 可 停 在 同一 排 五 个 连续 的 大 车 位 上 ， 但 不 能 停 在 小 车 位 上 。 
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在 下 面 的 实现 中 ， 我 们 创建 了 抽象 类 Vehicle， 而 car、Bus 和 Motorcycle 都 继承 自 这 个 类 。 
为 处 理 不 同 大 小 的 车 位 ， 我 们 用 了 一 个 类 Parkingspot， 并 以 它 的 成 员 变 量 表示 车 位 大 小 。 


public enum VehicleSize { Motorcycle, Compact, Large } 


1 

2 

3 public abstract class Vehicle { 

4 protected ArrayList<ParkingSpot> parkingSpots = 
5 new ArrayList<ParkingSpot>(); 

6 protected String licensePplate; 

7 protected int spotsNeeded; 

8 protected VehicleSize size; 

9 


16 public int getSpotsNeeded() { return spotsNeeded; } 
11 public VehicleSize getSize() { return size; } 


13 ”/* 将 车 辆 停 在 这 个 车 位 里 (也 可 能 包含 其 他 车 位 ) */ 
14 public void parkInSspot(Parkingspot s) { parkingSpots.add(s); } 


15 

16 /* 从 车 位 移 除 车 辆 ， 并 通知 车 位 车 辆 已 离开 */ 
17 public void clearSpots() { ... } 

18 


19 /* 检查 车 位 是 否 够 大 以 停放 该 车 辆 ( 且 车 位 是 空 的 ) ， 

26 * 这 只 会 检查 车 位 大 小 ， 并 不 检查 是 否 有 足够 多 

21 * 的 车 位 */ 

22 public abstract boolean canFitInSpot(ParkingSpot spot); 
23 0 


25 public class Bus extends Vehicle { 
26 public Bus() { 


27 spotsNeeded = 5; 

28 size = VehicleSize.Large; 

29 } 

36 

31 /* 检查 车 位 是 否 为 大 车 位 ， 不 会 检查 车 位 的 数目 */ 

32 public boolean canFitInSpot(ParkingSpot spot) { ... } 
33 } 

34 


35 public class Car extends Vehicle { 
36 public Car() { 


37 spotsNeeded = 1; 

38 size = VehicleSize.Compact; 

39 } 

46 

41 /* 检查 车 位 是 小 车 位 还 是 大 车 位 */ 

42 public boolean canFitInSpot(ParkingSpot spot) { ... } 
43 } 

44 


45 public class Motorcycle extends Vehicle { 
46 public Motorcycle() { 


47 spotsNeeded = 1; 

48 size = VehicleSize.Motorcycle; 
49 } 

56 
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51 public boolean canFitInSpot(ParkingSpot spot) { ... } 
52 } 


ParkingLot 类 本 质 上 就 是 Leve1 数 组 的 包 囊 类 。 以 这 种 方式 实现 ,我们 就 能 将 真正 寻找 空 车 
位 和 泪 车 的 处 理 逻 辑 从 ParkingLot 里 更 为 广泛 的 动作 中 抽取 出 来 。 要 是 不 这 么 做 ， 就 需要 将 车 
位 放 在 某 种 双 数 组 中 ( 或 将 车 位 位 于 所 在 楼 层 的 编号 对 应 到 车 位 列表 的 散 列 表 )。 将 ParkingLot 
与 Level 分 离开 来 ， 整 个 设计 更 显 清晰 。 








1 public class ParkingLot { 

private Level[] levels; 

3 private final int NUM LEVELS = 5; 
4 

5 public ParkingLot() { ... } 

6 

7 /* 将 该 车 辆 停 在 一 个 车 位 或 多 个 车 位 ， 
8 * 失败 则 返回 false */ 

9 public boolean parkVehicle(Vehicle vehicle) { ... } 
16 } 

11 


12 /* 代表 停车 场 里 的 一 层 */ 

13 public class Level { 

14 private int floor; 

15 private ParkingSpot[] spots; 

16 private int availableSpots = 6;j // 空闲 车 位 的 数量 
17 private static final int SPOTS_PER_ROW = 108; 


18 

19 public Level(int flr, int numberSpots) { ... } 

20 

21 public int availableSpots() { return availableSpots; } 
22 

23 /* 找 地 方 停 这 辆 车 ， 失 败 则 返回 false */ 

24 public boolean parkVehicle(Vehicle vehicle) { ... } 

25 

26 /* 停放 该 车 辆 ， 从 车 位 编号 spotNumber 开 始 ， 

27 * 直到 vehicle.spotsNeeded */ 

28 private boolean parkStartingAtSpot(int num, Vehicle v) { ... } 
29 

36 /* 寻找 车 位 停放 这 辆 车 。 返 回 车 位 索引 号 ， 

31 * 失败 则 返回 -1 */ 

32 private int findAvailableSpots(Vehicle vehicle) { ... } 
33 

34 /* 当 有 车 辆 从 车 位 移 除 时 ,增加 可 用 车 位 数 

35 * availableSpots */ 

36 public void spotFreed() { availableSpots++; } 

37 } 





Parkingspot 类 只 用 一 个 变量 表示 车 位 的 大 小 。 我 们 也 可 以 从 ParkingSpot 继 承 并 创建 
LargeSpot 、CompactSspot 和 MotorcycleSspot 等 几 个 类 来 实现 ,但 这 么 做 未 免 有 些小 题 大 做 。 除 
了 大 小 不 一 ， 这 些 车 位 并 没有 不 一 样 的 行为 。 


1 public class ParkingSpot { 
2 private Vehicle vehicle; 
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3 private VehicleSsize spotSize; 

4 private int row; 

5 private int spotNumber; 

6 private Level level; 

学 

8 public ParkingSpot(Level 1vl1l, int r, int n，VehicleSize s) {...} 
9 

16 public boolean isAvailable() { return vehicle == null; } 
11 

12 ”/* 检查 车 位 是 否 够 大 、 可 用 */ 

13 public boolean canFitVehicle(Vehicle vehicle) { ... } 

14 

15 /* 将 车 辆 停 在 该 车 位 */ 

16 public boolean park(Vehicle v) { ... } 

17 


18 public int getRow() { return row; } 
19 public int getSpotNumber() { return spotNumber; } 


21 /* 从 车 位 移 除 车 辆 ， 并 通知 楼 层 ， 

22 * 有 新 的 车 位 可 用 */ 

23 public void removeVehicle() { ... } 
24 } 


在 本 书 可 下 载 的 源码 包 中 ， 可 以 找到 上 述 代 码 的 完整 实现 ， 包 括 可 执行 的 测试 代码 。 
8.5 ”请 设计 在 线 图 书 阅读 器 系统 的 数据 结构 。( 第 66 页 ) 


解法 
此 题 对 系统 功能 的 说 明 着 墨 不 多 , 因此 ,就 让 我 们 假设 要 设计 一 个 基本 的 在 线 图 书 阅读 系统 ， 
提供 如 下 功能 。 





口 用 户 成 员 资 格 的 建立 和 延长 期 限 。 
口 搜索 图 书 数据 库 。 
口 阅读 书籍 。 
口 同一 时 间 只 能 有 一 个 活跃 用 户 。 
口 该 用 户 一 次 只 能 看 一 本 书 。 

要 实现 这 些 操 作 ， 可 能 还 需 提供 许多 其 他 函数 ， 比 如 get 、set 、update， 等 等 。 该 系统 的 
对 象 可 能 包括 User 、Book 和 Library。 

OnlineReadersystem 类 为 程序 的 主体 ， 可 以 这 么 实现 : 存放 所 有 图 书 的 信息 ， 管 理 用 户 ， 
刷新 显示 画面 ,但 是 这 么 一 来 ,整个 类 就 会 变 得 非常 笨重 。 因 此 , 我 们 转 而 选择 将 这 些 组 件 拆 分 
成 Library、 a 7 个 类 











1 public class OnlineReaderSystem { 

2 private Library library; 

3 private UserManager userManager; 
4 private Display display; 
5 
6 
学 


private Book activeBook; 
private User activeUser; 
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8 

9 public OnlineReaderSystem() { 

16 userManager = new UserManager(); 

1 library = new Library(); 

二 display = new Display(); 

13 } 

14 

15 public Library getLibrary() { return library; } 

16 public UserManager getUserManager() { return userManager; } 


17 public Display getDisplay() { return display; } 


19 public Book getActiveBook() { return activeBook; } 
26 public void setActiveBook(Book book) { 


于 activeBook = book 

22 display.displayBook(book); 

23 } 

24 

25 public User getActiveUser() { return activeUser; } 
26 public void setActiveUser(User user) { 

27 activeUser = user; 

28 display.displayUser(user); 

29 } 

36 } 


随后 ， 我 们 实现 这 几 个 类 ， 以 处 理 用 户 管理 器 、 图 书库 和 显示 组 件 。 








public class Library { 
private Hashtable<Integer, Book> books; 


1 
2 
3 
4 public Book addBook(int id, String details) { 
5 if (books.containsKey(id)) { 

6 return null; 

2 | 

8 Book book = new Book(id, details); 

9 books.put(id, book); 


16 Peturn book ; 

11 } 

12 

13 public boolean remove(Book b) { return remove(b.getID()); } 
14 public boolean remove(int id) { 

15 if (!books.containsKey(id)) { 

16 return false; 

17 } 

18 books .remove(id); 

19 return true; 

26 } 

21 

22 public Book find(int id) { 

23 return books.get(id); 

24 } 

25 } 

26 

27 public class UserManager { 

28 private Hashtable<Integer, User> users; 
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81 
82 


public User addUser(int id, String details, int accountType) { 
if (users.containsKey(id)) { 
return null; 
} 
User user = new User(id, details, accountType); 
users.put(id, user); 
return user; 


} 


public boolean remove(User u) { 
return remove(u.getID()); 


} 


public boolean remove(int id) { 
if (!users.containsKey(id)) { 
return false; 
} 
users.remove(id); 
return true; 


} 


public User find(int id) { 
return users.get(id); 


} 


public class Display { 
private Book activeBook; 
private User activeUser; 
private int pageNumber = 6) 


public void displayUser(User user) { 
activeUser = user; 
refreshUsername(); 


} 


public void displayBook(Book book) { 
pageNumber = 0) 
activeBook = book; 


refreshTitle(); 
refreshDetails(); 
refreshpage(); 


} 


public void turnPageForward() { 
pageNumber++; 
refreshpage(); 


} 


public void turnPageBackward() { 
pageNumber--; 
refreshpage(); 


图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 





206 第 9 章 解 题 技巧 





83 } 

84 

85 public void refreshUsername() { /* 更 新 用 户 名 的 显示 */ } 
86 public void refreshTitle() { /* 更 新 标题 的 显示 */ } 

87 public void refreshDetails() { /* 更 新 细节 信息 的 显示 */ 】》 
88 public void refreshPage() { /* 更 新 页 面 显 示 */ } 

89 } 


User 和 Book 类 只 是 存放 数据 ， 并 没有 什么 真正 的 功能 。 








public class Book { 
private int bookId; 
private String details; 


1 
2 
3 
4 
5 public Book(int id, String det) { 
6 bookId = id; 

7 details = det; 

8 } 

9 

16 public int getID() { return bookId; } 

11 public void setID(int id) { bookId = id; } 

12 public String getDetails() { return details; } 

13 public void setDetails(String d) { details = d; } 


14 } 

15 

16 public class User { 

17 private int userId; 

18 private String details; 

19 private int accountType; 

20 

21 public void renewMembership() { } 
22 

23 public User(int id, String details, int accountType) { 
24 userId = id; 

5 this.details = details; 

26 this.accountType = accountType; 
27 } 

28 


29 /* getter 和 setter */ 

36 public int getID() { return userId; } 

31 public void setID(int id) { userId = id; } 
32 public String getDetails() { 


33 return details; 

34 } 

35 

36 public void setDetails(String details) { 
37 this.details = details; 

38 } 


39 public int getAccountType() { return accountType; } 
46 public void setAccountType(int t) { accountType = t; } 
41 } 


用 户 管理 、 ee 等 功能 本 可 以 通通 放 进 on1ineReaderSystem 类 
它们 拆 分 至 不 同 的 类 里 ,这么 做 挺 有 意思 的 ,值得 探讨 一 番 。 如 果 一 个 系统 很 小 ， 
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使 系统 变 得 过 于 复杂 。 然 而 ， 随 着 系统 的 扩展 ，0nlineReaderSystem 会 加 入 越 来 越 多 的 功能 ， 


将 各 个 功 色 


6 拆 分 开 来 ， 可 以 避免 这 个 主 类 变 得 爱 肿 不 堪 。 





8.6 ”实现 一 个 拼图 程序 。 设 计 相 关 数 据 结构 并 提供 一 种 拼图 算法 。 假 设 你 有 一 个 fitswith 
方法 ， 传 入 两 块 拼 图 ， 若 两 块 拼图 能 拼 在 一 起 ， 则 返回 true。( 第 66 页 ) 


解法 


假设 有 一 套 传统 简单 的 拼图 游戏 , 按 行 和 列 划 分 为 网 格 , 每 块 拼 图 都 落 在 某 一 行 和 某 一 列 中 ， 


有 四 条 边 ， 


每 条 边 分 为 三 种 : 内 凹 、 外 凸 和 平 直 。 例 如 ， 角 落 的 拼图 块 有 两 条 边 是 平 直 的 ， 另 外 





两 条 边 可 能 是 内 凹 或 外 凸 。 














) ) 《 《 《 上 > 
< 和 全 起 > 起 by ao 个 




















在 玩 拼图 游戏 时 ( 手动 或 借助 算法 )， 我 们 需要 存储 每 块 拼图 的 位 置 ， 位 置 可 以 是 绝对 的 或 


相对 的 。 








口 绝对 位 置 :“ 这 块 拼 图 的 位 置 是 (12, 23) 。” 绝 对 位 置 属于 Piece 类 本 身 ， 同 时 还 包含 摆 放 
方向 。 
口 相对 位 置 :“ 我 不 知道 这 块 拼图 的 实际 位 置 ， 但 知道 它 与 另 一 块 拼图 相 邻 。” 相 对 位 置 属 








于 Edge 类 。 
我 们 的 解法 只 使 用 相对 位 置 ， 从 而 将 相 邻 的 边 拼 在 一 起 。 
下 面 是 一 种 可 能 的 面向 对 象 设 计 : 


1 
2 
3 
4 
5 
6 
这 
8 


9 
16 
1 
12 


class Edge { 


enum Type { inner, outer, flat } 
Piece parent; 

Type type; 

int index; // 指向 Piece.edges 的 索引 
Edge attached_to; // 相对 位 置 


/* 参见 算法 一 节 ， 若 两 块 拼图 应 该 拼 在 一 起 ， 
* 则 返回 true */ 
boolean fitsWwith(Edge edge) { ... }; 





13 class Piece { 


14 
15 


Edge[] edges; 
boolean isCorner() { ... } 
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18 class Puzzle { 
19 Piece[] pieces; /* 剩余 还 未 拼 的 拼图 */ 
26 Piece[][] solution; 


21 

22 /* 参见 算法 一 节 */ 

23 Edge[] inners, outers, flats; 
24 Piece[] corners; 

25 

26 /* 参见 算法 一 节 */ 

27 void sort() { ... } 

28 void solve() { ...} 

29 } 

拼 拼 图 的 算法 


下 面 我 们 将 搭配 使 用 伪 码 和 实际 代码 ， 勾 勒 出 拼 拼图 的 算法 。 








就 跟 小 孩 玩 拼图 游戏 时 一 样 , 我 们 会 从 最 简单 的 拼图 块 和 人手 : 四 个 角落 和 四 条 边 上 的 。 我 们 








很 容易 就 能 从 所 有 拼图 块 中 找 出 直 边 的 。 拼 图 的 时 候 , 不 妨 将 拼图 块 按 边缘 类 型 分 组 ,这 或 许 











个 不 错 的 选择 。 





1 void sort() { 

2 for each Piece p in pieces { 

3 if (p has two flat edges) then add p to corners 
4 for each edge in p.edges { 

5 if edge is inner then add to inners 

6 if edge is outer then add to outers 

7 } 

8 } 

9 } 





是 


如 此 一 来 ,给 定 某 一 边 , 我 们 可 以 更 快速 地 挑 出 可 能 拼合 在 一 起 的 拼图 块 。 然后 一 行 一 行 地 


人 


下 面 实现 的 solve 方 法 会 随意 挑选 一 个 角落 开始 拼图 ， 找 出 这 个 角落 还 没 拼 上 的 一 边 ， 然 后 


试 着 找 出 可 拼 在 一 起 的 拼图 块 。 找 到 相符 的 拼图 块 以 后 ， 执 行 如 下 操作 。 
(1) 与 边缘 衔接 起 来 。 
(2) 从 未 接 好 的 边缘 列表 中 移 除 该 边缘 。 
(3) 找到 下 一 条 未 接 好 的 边缘 。 


如 果 当 前 边缘 的 对 边 还 未 接 好 , 则 下 一 条 未 接 好 的 边缘 即 为 该 边缘 。 如 果 该 边缘 已 接 好 ， 


























下 一 条 边 可 以 是 任意 其 他 边缘 。 这 会 让 拼 拼图 时 看 起 来 像 是 从 外 向 内 的 螺旋 状 。 








则 


呈 螺 旋 状 的 原因 是 ， 只 要 可 以 的 话 ， 该 算法 总 是 以 直线 移动 。 当 抵达 第 一 边缘 的 末端 时 ， 算 
法 会 移 至 角落 拼图 块 唯一 可 用 的 边缘 , 也 就 是 旋转 90 度 。 每 到 边缘 的 末端 就 会 旋转 90 度 ， 直 到 拼 
图 外 圈 边 缘 全 部 拼 完 。 当 最 后 一 块 边缘 的 拼图 块 拼 好 后 ,该 拼图 块 只 剩 一 条 边 没 接 好 ， 于 是 再 次 

















旋转 90 度 。 在 后 续 每 一 圈 中 ， 该 算法 会 重复 同样 的 流程 ， 直 至 所 有 拼图 块 都 拼 好 为 目 。 
下 面 是 该 算法 的 类 似 Java 的 伪 码 实现 。 


图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 


9.8 面向 对 象 设计 


209 





52 
53 


public void solve() { 
/* 随便 选 个 角落 开始 拼图 */ 
Edge currentEdge = getExposedEdge(corner[6]); 


/# 循环 会 以 螺旋 状 进行 选 代 ， 
* 直到 拼图 完成 为 止 */ 
while (currentEdge != null) { 
/* 以 相反 的 边缘 类 型 进行 拼图 ， 内 媚 对 外 廿 ， 等 等 */ 
Edge[] opposites = currentEdge.type == inner ? 
outers : inners; 
for each Edge fittingEdge in opposites { 
if (currentEdge.fitswith(fittingEdge)) { 

attachEdges(currentEdge，fittingEdge); // 衔接 边缘 
removeFromList(currentEdge); 
removeFromList(fittingEdge); 


/* 取出 下 一 条 边缘 */ 
currentEdge = nextExposedEdge(fittingEdge); 
break; // 跳出 内 层 循环 ， 继 续 外 层 循 环 


} 


public void removeFromList(Edge edge) { 
if (edge.type == flat) return; 
Edge[] array = currentEdge.type == inner ? inners : outers; 
array.remove(edge); 


} 


/* 可 以 的 话 ， 返 回 对 边 的 边缘 ， 否 则 ， 
* 返回 任意 还 未 接 好 的 边缘 */ 

public Edge nextExposedEdge(Edge edge) { 
int next_index = (edge.index + 2) % 4; // 对 边 
Edge next edge = edge.parent.edges[next_ index]; 
if isExposed(next edge) { 

return next_edge; 

} 


return getExposedEdge(edge.parent); 


} 


public Edge attachEdges(Edge el, Edge e2) { 
el.attached to = e2; 
e2.attached to = el; 


} 


public Edge isExposed(Edge e1) { 
return edge.type != flat && edge.attached to == null; 
} 


public Edge getExposedEdge(Piece p) { 
for each Edge edge in p.edges { 
if (isExposed(edge)) { 
return edge; 
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55 } 

56 

57 return null; 
58 } 














为 了 简单 起 见 ， 我 们 将 inners 和 outers 表 示 为 一 个 Edge 数 组 。 但 这 并 不 是 个 好 设计 ， 因 为 
需要 频繁 添加 和 删除 数组 元 素 。 在 实际 代码 开发 中 ， 我 们 可 能 会 用 链表 来 实现 这 些 变量 。 

对 面试 来 说 ,要 写 出 此 题 的 完整 代码 ,实在 太 多 了 。 通常 ， 面 试 官 可 能 只 会 要 求 你 色 勒 代码 
的 轮 廊 。 


8.7 ”请 描述 该 如 何 设计 一 个 聊天 服务 器 。 要 求 给 出 各 种 后 台 组 件 、 类 和 方法 的 细节 ， 并 说 
明 其 中 最 难 解决 的 问题 会 是 什么 。( 第 66 页 ) 

解法 

设计 聊天 服务 器 是 项 大 工程 , 绝 非 一 次 面试 就 能 完成 。 毕 竞 ， 就 算 一 整个 团队 ,也 要 花费 数 
月 乃至 好 儿 年 才能 打造 出 一 个 聊天 服务 右 。 作 为 求职 者 ,你 的 工作 是 专注 解决 该 问题 的 某 个 方面 ， 
涉及 范围 要 够 广 ， 又 要 够 集中 ， 这 样 才能 在 一 轮 面试 中 搞定 。 它 不 一 定 要 与 真实 情况 一 模 一 样 ， 
但 也 应 该 忠实 反映 出 实际 的 实现 。 

这 里 我 们 会 把 注意 力 放 在 用 户 管理 和 对 话 等 核心 功能 : 添加 用 户 、 创 建 对 话 、 更 新 状态 ， 等 
等 。 考虑 到 时 间 和 空间 有 限 , 我 们 不 会 探讨 这 个 问题 的 联网 部 分 ， 也 不 描述 数据 是 怎么 真正 推送 
到 客户 端的 。 

男 外 ,我 们 假设 “好 友 关 系 ”是 双向 的 ， 如 有 果 你 是 我 的 联系 人 之 一 ， 那 就 表示 我 也 是 你 的 联 
系 人 之 一 。 我们 的 聊天 系统 将 支持 群 组 聊天 和 一 对 一 (私密 ) 聊天 , 但 并 不 考虑 语音 聊天 、 视 频 
聊天 或 文件 传输 。 


1. 需要 支持 哪些 特定 动作 ? 

这 也 有 待 你 跟 面 试 官 探 讨 ， 下 面 列 出 几 点 想法 。 

口 显示 在 线 和 离线 状态 。 

口 添加 请 求 ( 发 送 、 接 受 、 拒 绝 )。 

口 更 新 状态 信息 。 

口 发 起 私 聊 和 群 聊 。 

口 在 私 聊 和 群 聊 中 添加 新 信息 。 

这 只 是 一 部 分 列表 ， 如 果 时 间 有 富余 ， 还 可 以 多 加 一 些 动作 。 


2. 从 这 些 需求 可 了 解 到 什么 ? 
我 们 必须 掌握 用 户 、 添 加 请 求 的 状态 、 在 线 状态 和 消息 等 概念 。 


3. 系统 有 哪些 核心 组 件 ? 

这 个 系统 可 能 由 一 个 数据 库 、 一 组 客户 端 和 一 组 服务 器 组 成 。 我 们 的 面向 对 象 设 计 不 会 包含 
这 些 部 分 ， 不 过 可 以 讨论 一 下 系统 的 整体 概览 。 

数据 库 将 用 来 存放 更 持久 的 数据 ， 比 如 用 户 列 表 或 聊天 对 话 的 备份 。SQL 数 据 库 应 该 是 不 错 
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的 选择 ， 或 者 ， 如 果 可 扩展 性 要 求 更 高 ， 可 以 选用 BigTable 或 其 他 类 似 的 系统 。 

对 于 客户 端 和 服务 器 之 间 的 通信 ， 使 用 XML 应 该 也 不 错 。 尽 管 这 种 格式 不 是 最 紧凑 的 (你 
也 应 该 向 面试 官 指出 这 一 点 )， 它 仍 是 很 不 错 的 选择 ， 因 为 不 管 是 计算 机 还 是 人 类 都 容易 辨识 。 
使 用 XML 可 以 让 程序 调试 起 来 更 轻松 ， 这 一 点 非常 重要 。 

服务 器 由 一 组 机 器 组 成 ， 数 据 会 分 散 到 各 台 机 器 上 , 这 样 一 来 ,我们 可 能 就 必须 从 一 台 机 器 
跳 到 另 一 台 机 器 。 如 果 可 能 的 话 , 我们 会 尽量 在 所 有 机 器 上 复制 部 分 数据 ,以 减少 查询 操作 的 次 
数 。 在 此 , 设计 上 有 个 重要 的 限制 条 件 ， 就 是 必须 防止 出 现 单 点 故障 。 例 如 ， 如 果 一 台 机 器 控制 
所 有 用 户 的 登录 ， 那 么 ， 只 要 这 一 台 机 器 断 网 ， 就 会 造成 数 以 百 万 计 的 用 户 无 法 登录 。 


4. 有 哪些 关键 的 对 象 和 方法 ? 
系统 的 关键 对 象 包 括 用 户 、 对 话 和 状态 消息 等 ， 我 们 已 经 实现 了 UserManagement 类 。 要 是 
更 关注 这 个 问题 的 联网 方面 或 其 他 组 件 ， 我 们 就 可 能 转 而 深入 探究 那些 对 象 。 


1 /* UserManager 用 作 核 心 用 户 动作 的 控制 中 心 */ 
2 public class UserManager { 
3 private static UserManager instance; 

































































4 /* 从 用 户 识别 码 映 射 到 用 户 */ 

5 private HashMap<Integer, User> UsersById ; 

6 

7 /* 从 帐户 名 映射 到 用 户 */ 

8 private HashMap<String, User> usersByAccountName; 

9 

16 /* 从 用 户 识别 码 映 射 到 在线 用 户 */ 

11 private HashMap<Integer, User> onlineUsers; 

12 

13 public static UserManager getInstance() { 

14 if (instance == null) instance = new UserManager(); 
15 return instance; 

16 } 

17 

18 public void addUser(User fromUser, String toAccountName) { ... } 
19 public void approveAddRequest(AddRequest req) { ... } 
16 public void rejectAddRequest(AddRequest req) { ... } 
21 public void userSignedon(String accountName) { ... } 
22 public void userSignedoff(String accountName) { ... } 
23 } 


在 User 类 中 ，receivedAddRequest 方 法 会 通知 用 户 B (User B )， 用 户 A (User A ) 请 求 加 
他 为 好 友 。 用 户 B 会 接受 或 拒绝 该 请 求 (通过 UserManager.approveAddRequest 或 
rejectAddRequest )，UserManager 则 负责 将 用 户 互相 添加 到 对 方 的 通讯 录 中 。 

当 UserManager 要 将 AddRequest 加 入 用 户 A 的 请 求 列表 时 ,会 调用 User 类 的 sentAddRequest 
方法 。 综 上 ， 整 个 流程 如 下 。 

(1) 用 户 A 点 击 客户 端 软件 上 的 “添加 用 户 ”， 发 送 给 服务 器 。 

(2) 用 户 A 调 用 requestAddUser(User B)。 

(3) 步骤 2 的 方法 会 调用 UserManager .addUser。 

(4) UserManager 会 调用 User A.sentAddRequest 和 User B.receivedAddRequest。 
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第 9 章 





重申 一 下 ,这 只 是 设计 这 些 交 互 的 其 中 一 种 方式 ,但 这 不 是 


的 做 法 。 


1 
2 
3 
4 
5 
6 
pe 
8 
9 


37 
38 



































瑟 


作 一 的 方式 ,其 至 也 不 是 唯一 “好 ” 











public class User { 
private int id; 
private UserStatus status = null; 


} 


/* 将 其 他 参与 的 用 户 识别 码 映射 到 对 话 */ 
private HashMap<Integer, PrivateChat> privateChats; 


/* 将 群 聊 识别 码 映射 到 群 聊 */ 
private ArrayList<GroupChat> groupChats; 


/* 将 其 他 人 的 用 户 识别 码 映 射 到 加 入 请 求 */ 
private HashMap<Integer, AddRequest> receivedAddRequests; 


/* 将 其 他 人 的 用 户 识别 码 映射 到 加 入 请 求 */ 
private HashMap<Integer, AddRequest> sentAddRequests; 








/* 将 用 户 识 别 码 映射 到 加 入 请 求 */ 
private HashMap<Integer, User> contacts; 


private String accountName; 
private String fullName; 


public 
public 
public 
public 
public 
public 
public 
public 
public 
public 
public 
public 
public 
public 
public 


User(int id, String accountName, String fullName) { ... } 
boolean sendMessageToUser(User to, String content){ ... } 
boolean sendMessageToGroupChat(int id, String cnt){...} 
void setStatus(UserStatus status) { ... } 

UserStatus getStatus() { ... } 

boolean addContact(User user) { ... } 

void receivedAddRequest(AddRequest req) { ...} 

void sentAddRequest(AddRequest req) { ... } 

void removeAddRequest(AddRequest req) { ... } 

void requestAddUser(String accountName) { ... } 

void addConversation(PrivateChat conversation) { ... } 
void addConversation(GroupChat conversation) { ... } 

int getId() { ... } 

String getAccountName() { ... } 

String getFullName() { ... } 





Conversation 类 实现 为 一 个 抽象 类 ， 为 所 有 Conversation 不 是 GroupChat 就 是 
PrivateChat ， 同 时 每 个 类 各 有 自己 的 功能 。 


1 
2 
3 
4 
5 
6 
2 
8 
9 


16 


public abstract class Conversation { 
protected ArrayList<User> participants; 
protected int id; 

protected ArrayList<Message> messages; 


} 


public 
public 
public 


ArrayList<Message> getMessages() { ... } 
boolean addMessage(Message m) { ... } 
int getId() { ... } 
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11 public class GroupChat extends Conversation { 


12 public void removeParticipant(User user) { ... } 
13 public void addParticipant(User user) { ... } 
14 } 

15 

16 public class PrivateChat extends Conversation { 

17 public PrivateChat(User user1, User user2) { ... 
18 public User getotherParticipant(User primary) { ... } 
19 } 

20 

21 public class Message { 

22 private String content; 

23 private Date date,; 

24 public Message(String content, Date date) { ... } 
25 public String getContent() { ... } 

26 public Date getDate() { ... } 

27 } 


AddRequest 和 Userstatus 两 个 类 比较 简单 ， 功 能 不 多 ， 主 要 用 来 将 数据 聚合 在 一 起 ， 方 便 
其 他 类 使 用 。 





1 public class AddRequest { 

2 private User fromUser; 

3 private User toUser; 

4 private Date date; 

5 Requeststatus status; 

6 

7 public AddRequest(User from, User to, Date date) { ... } 
8 public RequestStatus getStatus() { ... } 

9 public User getFromUser() { ... } 

16 public User getToUser() { ... } 

11 public Date getDate() { ... } 

12 } 

13 

14 public class UserStatus { 

15. private String message; 

16 private UserStatusType type; 

17 public UserStatus(UserStatusType type, String message) { ... } 
18 public UserStatusType getStatusType() { ... } 
19 public String getMessage() { ... } 

26 } 

21 


22 public enum UserStatusType { 
23 Offline, Away, Idle, Available, Busy 


26 public enum RequestStatus { 
27 Unread, Read, Accepted, Rejected 
28 } 


在 本 书 可 下 载 的 完整 源码 中 ， 可 以 查看 这 些 方法 的 更 多 细节 ， 包 括 上 述 方法 的 具体 实现 。 


5. 最 难 解 决 或 最 有 意思 的 问题 是 什么 ? 
下 面 这 些 问题 可 能 有 点 意思 ， 不 妨 与 面试 官 深 入 探讨 一 看 。 
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问题 1: 如 何 确定 菜 人 在 线 一 一 我 指 的 是 真 的 、 真 的 知道 ? 
虽然 希望 用 户 在 退出 时 通知 我 们 , 但 即便 如 此 也 无 法 确切 知道 状态 。 例如， 用 户 的 网 络 连 接 
可 能 断 开 了 。 为 了 确定 用 户 何 时 退出 ， 或 许可 以 试 着 定期 询问 客户 端 ， 以 确保 它 仍 然 在 线 。 























问题 2: 如 何 处 理 冲 突 的 信息 ? 
部 分 信息 存储 在 计算 机 内 存 中 ,部 分 则 存储 在 数据 库 里 。 如 果 两 者 不 同步 有 冲突 ， 那 会 出 什 
么 问题 ? 哪 一 部 分 是 “正确 的 ”? 





问题 3: 如 何 才能 让 服务 器 在 任何 负载 下 都 能 应 付 自如 ? 
前 面 我 们 设计 聊天 服务 器 时 并 没 怎么 考虑 可 扩展 性 , 但 在 实际 场景 中 必须 予以 关注 。 我 们 需 
要 将 数据 分 散 到 多 台 服 务 器 上 ， 而 这 又 要 求 我 们 更 关注 数据 的 不 同步 。 





问题 4: 如 何 预防 拒绝 服务 攻击 ? 
客户 端 可 以 向 我 们 推送 数据 一 一 若 它 们 试图 向 服务 需 发 起 拒绝 服务 (DOS ) 攻击 ， 怎 么 办 ? 
该 如 何 预防 ? 


8.8 “奥赛 罗 棋 ”( 黑白 棋 ) 的 玩法 如 下 : 每 一 枚 棋子 的 一 面 为 白 ， 一 面 为 黑 。 游 戏 双 方 各 
执 黑 、 白 棋子 对 决 ， 当 一 枚 棋子 的 左右 或 上 下 同时 被 对 方 棋子 夹 住 ， 这 枚 棋子 就 算是 被 吃 掉 了 ， 
随即 翻 面 为 对 方 棋子 的 颜色 。 轮 到 你 落 子 时 , 必须 至 少 吃 掉 对 方 一 枚 棋子 。 任意 一 方 无 子 可 落 时 ， 
游戏 即 告 结束 。 最 后 , 棋盘 上 棋子 较 多 的 一 方 获胜 。 请 运用 面向 对 象 设计 方法 , 实现 “奥赛 罗 棋 ”。 
(第 66 页 ) 

解法 

我 们 先 来 举 个 例子 。 假 设 在 一 盘 奥 赛 罗 棋 中 ， 有 如 下 棋 步 。 

(1) 初始 化 棋盘 ， 在 中 心 位 置 布下 两 枚 黑子 和 两 枚 白 子 。 两 枚 黑子 分 别 落 在 中 心 点 的 左上 方 
和 右 下 方 。 

(2) 在 6 行 4 列 处 落 黑 子 ， 则 $ 行 4 列 的 白 子 翻 面 变 为 黑子 。 

(3) 在 4 行 3 列 处 落 白 子 ， 则 4 行 4 列 的 黑子 翻 面 变 为 白 子 。 

经 过 上 面 的 棋 步 ， 棋 盘 布 局 如 下 。 
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在 奥赛 罗 棋 中 ， 核 心 对 象 大 致 有 游戏 ( game )、 棋 盘 (board )、 棋 子 (piece， 黑 子 或 白 子 ) 
和 玩家 (player )。 该 如 何 用 面向 对 象 设 计 优雅 地 表示 这 些 对 象 ? 


1. 该 不 该 创建 BlackPiece 和 WhitePiece 类 ? 

起 先 ， 我 们 可 能 认为 自己 需要 从 Piece 抽 象 类 派生 出 Blackpiece 类 和 whitepiece 类 。 然 而 ， 
这 么 做 不 见得 好 。 每 颗 模 子 都 可 以 来 回 翻 面 ， 黑 变 白 ， 白 变 黑 ， 这 么 来 看 ， 连 绪 不 断 地 销 席 和 创 
建 完全 相同 的 对 象 并 不 明智 。 因 此 ， 更 好 的 做 法 可 能 是 只 创建 Piece 类 ， 并 用 标记 指示 棋子 当前 
的 颜色 。 


2. 需要 Board 和 Game 两 个 独立 的 类 吗 ? 

严格 来 说 ， 可 能 没有 必要 有 既 创 建 Game 对 象 又 引入 Board 对 象 。 不 过 ， 分 别 创建 这 两 个 对 象 可 
以 从 人 逻辑 上 划分 棋盘 ( 只 含 涉及 沙子 的 逻辑 处 理 ) 和 游戏 ( 含 计时 、 游 戏 流程 等 )。 但 是 这 么 做 
也 有 次 端 , 我 们 的 程序 会 多 加 几 层 处 理 ， 变 得 更 复杂 。 有 个 函数 可 能 会 调用 Game 的 方法 , 却 只 是 
为 了 让 它 去 调用 Board 里 的 方法 。 下 面 我 们 决定 将 Game 和 Board 分 开创 建 ， 不 过 面试 时 最 好 跟 面 
试 官 讨论 一 下 。 


3. 谁 来 记录 分 数 ? 

很 显然 , 我 们 需要 某 种 记分 方式 来 记录 黑子 和 白 子 的 数目 。 但 该 由 程序 的 哪 部 分 来 负责 维护 
这 些 信息 ? 不 管 是 由 Game 抑 或 Board 甚 至 由 Piece (在 静态 方法 中 ) 维护 这 些 信 息 ， 各 有 各 的 理 
由 。 我 们 选择 交 由 Board 保 存 这 部 分 信息 , 分 数 在 逻辑 上 可 以 算是 棋盘 的 一 部 分 , 由 Piece 或 Board 
调用 Board 类 的 colorChanged 和 colorAdded 方 法 进行 更 新 。 




































































4. Game 该 不 该 实现 成 单 态 类 ? 

将 Game 实 现 为 单 态 类 ， 优 点 在 于 Game 的 方法 调用 起 来 很 容易 ， 不 用 将 Game 对 象 的 引用 传 来 
传 去 。 

不 过 ， 将 Game 实 现成 单 态 类 也 意味 着 它 只 能 实例 化 一 次 ， 这 个 假设 条 件 成 立 吗 ? 在 面试 时 ， 
最 好 与 面试 官 交流 一 下 。 

下 面 是 奥赛 罗 棋 的 一 种 可 能 设计 。 
public enum Direction { 


left, right, up, down 
} 





























public enum Color { 
White, Black 


} 


ovOmA 和 Fw 哺 


9 public class Game { 

16 private Player[] players; 

11 private static Game instance; 
12 private Board board; 

13 private final int ROWS = 16; 

14 private final int COLUMNS = 16; 
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15 

16 private Game() { 

17 board = new Board(ROWS, COLUMNS); 

18 players = new Player[2]; 

19 players[6] = new Player(Color.Black); 
26 players[1] = new Player(Color.White); 
21 } 

22 

23 public static Game getInstance() { 

24 if (instance == null) instance = new Game(); 
25 return instance; 

26 } 

27 

28 public Board getBoard() { 

29 return board; 

36 } 

31 } 

















Board 类 负责 管理 棋子 本 身 ， 但 并 不 处 理 游戏 玩法 的 部 分 ， 而 是 交 由 Game 类 处 理 。 








1 public class Board { 

2 private int blackCount = 0@; 

3 private int whiteCount = 0@; 

4 private Piece[][] board; 

3 

6 public Board(int rows, int columns) { 

7 board = new Piece[rows][columns]; 

8 } 

9 

16 public void initialize() { 

11 /* 初始 化 棋盘 中 心 的 白 子 和 黑子 */ 

12 } 

13 

14 /* 试 着 将 颜色 为 Color 的 棋子 放 在 (row，column) 位 置 
15 * 成 功 则 返回 true */ 

16 public boolean placeColor(int row, int column, Color color) { 
17 ce 

18 } 

19 


26 /* 从 (row，column) 开 始 ， 顺 着 方向 d， 
这 * 将 棋子 翻 面 */ 


22 private int flipSection(int row, int column, Color color， 
23 Direction d) { ... } 

24 

25 public int getScoreForColor(Color c) { 

26 if (c == Color.Black) return blackCount; 

27 else return whiteCount; 

28 } 

29 


36 /* 更 新 棋盘 ， 有 newPieces 个 棋子 变 为 newCoLor 颜 色 ， 
31 * 减少 另 一 种 颜色 的 分 数 */ 
32 public void updateScore(Color newColor, int newPieces) { ... } 
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如 前 所 述 ， 我 们 会 用 Piece 类 实现 黑白 棋子 ， 该 类 有 个 简单 的 Color 变 量 ， 表 示 棋 子 是 黑子 
还 是 白 于 

















1 public class Piece { 

2 private Color color; 

3 public Piece(Color c) { color = cj } 

4 

5 public void flip() { 

6 if (color == Color.Black) color = Color.White; 
7 else color = Color.Black; 

8 } 

9 

16 public Color getColor() { return color; } 
11 } 





Player 存 放 的 信息 非常 有 限 ， 甚 至 不 会 保存 自己 的 分 数 ， 但 有 个 方法 可 用 来 获取 分 数 。 
Player.getscore() 会 调用 GameManager 取 得 分 数 。 


12 public class Player { 


13 private Color color; 

14 public Player(Color c) { color = c; } 

15 

16 public int getScore() { ... } 

17 

18 public boolean playPiece(int r, int c) { 
19 return Game.getInstance().getBoard().placeColor(r, c, color); 
26 } 

21 

22 public Color getColor() { return color; } 
23 } 


本 书 可 下 载 的 源码 包 提 供 了 完整 可 运行 的 版 本 。 

记 住 ,在 处 理 很 多 问题 时 ， 相 比 你 做 了 些 什 么 ,你 为 什么 这 么 做 反而 更 显 重要 。 面 试 官 也 许 
不 会 在 意 你 是 否 选择 将 Game 类 实现 为 单 态 类 , 但 她 可 能 真 的 在 乎 你 有 没有 花 时 间 思 考 , 有 没有 跟 
她 讨论 各 种 做 法 的 优 劣 。 


8.9 设计 一 种 内 存 文件 系统 (in-memory file system ) 的 数据 结构 和 算法 ， 并 说 明 具 体 
做 法 。 如 有 可 行 ， 请 用 代码 举例 说 明 。( 第 66 页 ) 


解法 

许多 求职 者 一 看 到 这 个 问题 ， 可 能 就 会 惊慌 失措 。 文 件 系 统 太 底层 了 吧 ! 

其 实 , 没 必要 惊慌 。 只 要 把 文件 系统 的 组 件 考 虑 周全 , 我 们 就 能 像 解 决 其 他 面向 对 象 设计 问 
题 那样 搞定 此 题 。 

一 个 最 简单 的 文件 系统 由 File (文件 ) 和 Directory ( 目录 ) 组 成 。 每 个 Directory 包 含 一 
组 File 和 Directory。File 和 Directory 有 很 多 相同 特征 ， 因 此 我 们 创建 了 Entry 类 ， 前 面 两 个 
类 则 继承 这 个 类 。 


1 public abstract class Entry { 9 
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之 protected Directory parent; 

3 protected long created; 

4 protected long lastUpdated; 

5 protected long lastAccessed; 

6 protected String name; 

水 

8 public Entry(String n, Directory p) { 

9 name = nj 

16 parent = p; 

created = System.currentTimeMillis(); 

12 lastUpdated = System.currentTimeMillis(); 
13 lastAccessed = System.currentTimeMillis(); 
14 } 

15 

16 public boolean delete() { 

17 if (parent == null) return false; 

18 return parent.deleteEntry(this); 

19 } 

20 

21 public abstract int size(); 

22 

23 public String getFullpath() { 

24 if (parent == null) return name; 

25 else return parent.getFullpath() + “/” + name; 
26 } 

27 


28 /* getter 和 setter */ 

29 public long getCreationTime() { return created; } 

36 public long getLastUpdatedTime() { return lastUpdated; } 
31 public long getLastAccessedTime() { return lastAccessed; } 


32 public void changeName(String n) { name = nj } 
33 public String getName() { return name; } 

34 } 

35 

36 public class File extends Entry { 

37 private String content; 

38 private int size; 

39 

40 public File(String n, Directory p, int sz) { 
41 super(n, p); 

42 size = sz; 

43 } 

44 

45 public int size() { return size; } 


46 public String getContents() { return content; } 

47 public void setContents(String c) { content = c; } 
48 } 

49 

56 public class Directory extends Entry { 

51 protected ArrayList<Entry> contents; 


52 

53 public Directory(String n, Directory p) { 
54 super(n, p); 

55 contents = new ArrayList<Entry>(); 
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56 } 

57 

58 public int size() { 

59 int size = 0; 

66 for (Entry e : contents) { 

61 size += e.size(); 

62 } 

63 return size; 

64 } 

65 

66 public int numberOfFiles() { 

67 int count = 0; 

68 for (Entry e : contents) { 

69 if (e instanceof Directory) { 
76 count++; // 目录 也 算 作 文件 

71 Directory d = (Directory) e; 
72 count += d.numberOfFiles(); 
73 } else if (e instanceof File) { 
74 count++; 

75 } 

76 } 

77 return count; 

78 } 

79 

80 public boolean deleteEntry(Entry entry) { 
81 return contents.remove(entry); 

82 } 

83 

84 public void addEntry(Entry entry) { 
85 contents.add(entry); 

86 } 

87 

88 protected ArrayList<Entry> getContents() { return contents; } 
89 } 


另外 ， 我 们 还 可 以 这 样 实现 Directory: 为 文件 和 子 目 录 创 建 不 同 的 链表 。 如 此 一 来 ， 
numberofFiles() 方 法 就 不 需要 再 用 instanceof 运 算 符 ， 所 以 更 为 简洁 , 不 过 , 我们 就 无 法 轻易 
按 日 期 或 名 称 对 文件 和 目录 进行 排序 。 

8.10 ”设计 并 实现 一 个 散 列 表 ， 使 用 链接 ( 即 链表 ) 处 理 磁 撞 冲突 。( 第 66 页 ) 

解法 

假设 我 们 要 实现 类 似 Hash<k，V> 的 散 列 表 。 即 ， 该 散 列 表 将 类 型 k< 的 对 象 映射 为 类 型 Vv 的 


对 象 。 
首先 ， 我 们 或 许 会 想到 数据 结构 应 该 大 致 如 下 : 





1 public class Hash<K，V> { 

2 LinkedList<V>[] items; 

3 public void put(K key, V value) { ... } 
4 public V get(K key) { ... } 

5 } 
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注意 ,items 是 个 链表 的 数组 ,其 中 items[i] 是 个 链表 ， 
即 在 i 处 碰撞 冲突 的 所 有 对 象 )。 














包含 所 有 键 映射 成 索引 i 的 对 象 ( 也 





这 么 做 看 似 可 行 ， 不 过 要 下 定论 ， 还 得 更 深入 一 些 考虑 碰撞 冲突 的 情 








假设 我 们 有 个 非常 简单 、 使 用 字符 串 长 度 的 散 列 函 数 。 


1 public int hashCodeOfKey(K key) { 
2 return key.toString().length() % items.length; 
3 } 





键入 m 和 bob 都 会 对 应 到 数组 的 同一 索引 ， 尺 管 这 两 个 键 并 不 一 样 。 我 们 必须 搜索 整个 链表 ， 























找 出 这 些 键 对 应 的 真正 对 象 , 但 是 该 怎么 办 呢 ? ee 





这 就 是 要 把 值 和 原先 的 键 一 并 存储 起 来 的 原因 。 








一 种 做 法 是 引入 一 个 Cel1 对 象 ， 存 储 键 值 对 。 在 这 种 实现 中 ， 链 表 元 素 的 类 型 为 Cel1。 


下 面 是 该 实现 的 代码 。 

public class Hash<K，V> { 
private final int MAX_SIZE = 10; 
LinkedList<Cell<k, V>>[] items; 
public Hash() { 


} 


9 /* 非常 非常 粗 陋 的 散 列 */ 
16 public int hashCodeofKey(K key) { 


1 
2 
3 
4 
5 
6 
2 
8 


11 return key.toSstring().length() % items.length; 
12 } 

13 

14 public void put(K key, V value) { 

15 int x = hashCodeOfKey (key); 

16 if (items[x] == null) { 

17 items[x] = new LinkedList<Cell<Kk, V>>(); 
18 

19 

26 LinkedList<Cell<K，V>> collided = items[x]; 
21 

22 /* 查找 有 着 相同 键 的 项 目 ， 若 找到 则 替换 掉 */ 

23 for (Cell<k, V> c : collided) { 

24 if (c.equivalent(key)) { 

25 collided.remove(c); 

26 break ; 

27 } 

28 } 

29 

36 Cell<K，V> cell = new Cell<K，V>(key，value); 
31 collided.add(cell); 

32 } 

33 

34 public V get(K key) { 

35 int x = hashCodeOfKey(key ); 

36 if (items[x] == null) { 
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37 return null; 

38 } 

39 LinkedList<Cell<K, V>> collided = items[x]; 
46 for (Cell<K，V> c : collided) { 
41 if (c.equivalent(key)) { 

42 return c.getValue(); 

43 } 

44 } 

45 

46 return null; 

47 } 

48 } 


Cell 类 存储 有 一 对 数据 值 和 键 。 这 样 一 来 ,我 们 就 可 以 搜索 整个 链表 ( 因 碰 撞 冲 突 而 建 , 但 
键 不 一 样 )， 找 到 对 应 该 键 值 的 对 象 。 





1 public class Cell<k, V> { 

2 private K key; 

3 private V value; 

4 public Cell(K k, Vv){ 

5 key = k; 

6 value = Vi 

7 } 

8 

9 public boolean equivalent(Cell<K, V> c) { 
16 return equivalent(c.getKey() ); 
11 } 

12 

13 public boolean equivalent(K k) { 
14 return key.equals(k); 

15 } 

16 


17 public K getKey() { return key; } 
18 public V getValue() { return value; } 





sa» 


也 


实现 散 列表 的 另 一 种 常见 做 法 是 使 用 二 又 查找 树 作为 底层 数据 结构 。 检 索 元 素 的 时 间 复 杂 度 
不 再 是 O(1) (不 过 ， 从 技术 上 来 说 ,复杂 度 不 会 是 O(1)， 因 为 可 能 有 很 多 碰撞 冲突 )， 但 是 
做 法 不 需要 创建 一 个 无 谓 的 大 数组 ， 用 以 存储 项 目 。 


9.9 递归 和 动态 规划 








站 








9.1 ”有 个 小 孩 正在 上 楼 梯 ， 楼 梯 有 n 阶 人 台阶， 小 孩 一 次 可 以 上 1 阶 、2 阶 或 3 阶 。 实 现 一 个 
方法 ， 计 算 小 孩 有 多 少 种 上 楼 梯 的 方式 。( 第 68 页 ) 


解法 
我 们 可 以 采用 自 上 而 下 的 方式 来 解决 这 个 问题 。 小 孩 上 楼 梯 的 最 后 一 步 ， 也 就 是 抵达 第 n 阶 
的 那 一 步 ， 可 能 走 1 阶 、2 阶 或 3 阶 。 也 就 是 说 ， 最 后 一 步 可 能 是 从 第 n-1 阶 往 上 走 1 阶 、 从 第 n-2 
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阶 往 上 走 2 阶 ， 或 从 第 n-3 阶 往 上 走 3 阶 。 因 此 ， 抵 达 最 后 一 阶 的 走 法 ， 其 实 就 是 抵达 这 最 后 三 阶 
的 方式 的 总 和 。 
下 面 是 该 算法 的 简单 实现 。 





1 public int countWays(int n) { 

2 if (n < 6) { 

3 return 0; 

4 } else if (n == 6) { 

5 return 1; 

6 } else { 

过 return countWays(n - 1) + countWays(n - 2) + 
8 countWays(n - 3); 

9 } 


16 } 

跟 韭 波 那 契 数 列 问 题 一 样 ， 这 个 算法 的 运行 时 间 呈 指数 级 增长 ( 准确 地 说 是 0(3”) )， 因 为 每 
次 调用 都 会 分 支出 三 次 调用 。 这 就 意味 着 ， 对 同一 数值 ，countWays 会 调用 很 多 次 ， 而 这 显然 没 
有 必要 。 我 们 可 以 利用 动态 规划 加 以 修正 。 











1 public static int countWaysDP(int n, int[] map) { 
2 if (n < 6) 1{ 

3 return 6 

4 } else if (n == 6) { 

5 return 1; 

6 } else if (map[n] > -1) { 

7 return map[n]; 

8 } else { 

9 map[n] = countWaysDP(n - 1, map) + 
16 countWaysDP(n - 2, map) + 
11 countWaysDP(n - 3, map); 
了 return map[n]; 

13 } 








无 论 是 否 使 用 动态 规划 , 注意 上 楼 梯 的 方式 总 数 很 快 就 会 突破 整数 ( int 型 ) 的 上 限 而 溢出 。 
当 n = 37 时 ， 结 果 就 会 溢出 。 使 用 long 可 以 撑 和 久 一 点 ， 但 也 不 能 从 根本 上 解决 问题 。 


9.2 设想 有 个 机 器 人 坐 在 XxY 网 格 的 左上 和 角 , 只 能 向 右 、 向 下 移动 。 机 器 人 从 (0,0) 到 (X,Y) 
有 多 少 种 走 法 ? 

进 阶 

假设 有 些 点 为 “禁区 ”， 机 器 人 不 能 踏足 。 设 计 一 种 算法 ， 找 出 一 条 路 径 ， 让 机 器 人 从 左上 
角 移 动 到 右 下 角 。( 第 68 页 ) 





解法 
我 们 需要 数 一 数 机 器 人 向 右 X 步 、 向 下 7 步 ， 总 共 可 以 走出 多 少 种 路 径 。 这 条 路 径 总 共有 
X+7Y 步 。 


为 了 走出 一 条 路 径 ， 我 们 实质 上 要 从 针 7 步 里 ， 选 出 X 沙 为 向 右 移动 。 因 此 ， 可 能 路 径 的 总 
数 就 是 从 X+7 项 中 选 出 X 项 的 方法 总 数 。 具 体 可 以 用 下 面 的 二 项 式 (又 称 “n 选 /”) 表示 : 
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及 | nl 
[= rl(n—r)! 


失地 | (X+7)! 
xX) XI7 


对 这 个 问题 来 说 ， 该 算式 变 成 : 





就 算 不 知道 二 项 式 ， 你 也 可 以 自行 推导 出 解法 。 

我 们 可 以 将 每 条 路 径 看 作 一 个 长 度 为 X+7 的 字符 串 ， 由 个 R 和 Y 个 D 组 成 。X+Y 个 不 同 的 字符 
可 以 组 成 (X+)! 个 字符 串 。 不 过 ， 在 这 个 问题 中 ， 有 XX 个 字符 为 R，7Y 个 字符 为 0D。R 有 了 种 排列 组 
合 ， 这 些 组 合 全 都 一 样 ， 对 5 的 情况 也 类 似 ， 因 此 必须 将 结果 除 以 如 和 刺 。 最 后 可 得 到 跟前 面相 
同 的 算式 : 





(X+Y)! 
XIY! 





进 阶 ， 找到 一 条 避 开 禁区 点 的 路 径 

如 果 把 网 格 画 出 来 ， 你 会 发 现 移 动 到 位 置 C&, 有 的 唯一 方式 ， 就 是 先 移动 到 它 的 相 邻 点 : 
(X-1,D 或 ,了 1)。 因 此 ， 我 们 需要 找到 一 条 移 至 (X-1,D 或 CY, 关 1) 的 路 径 。 

怎么 才能 找 出 前 往 这 些 位 置 的 路 径 呢 ?” 要 找 出 前 往 (X-1,D 或 (%, 关 1) 的 路 径 ， 我 们 需要 先 移 
至 其 中 一 个 相 邻 点 。 因 此 ,要 找到 一 条 路 径 移动 到 (X=1, 站 的 相 邻 点 ， 坐 标 为 (X-2, 六 和 (X-1, 关 1)， 
或 C 闫 1) 的 相 邻 点 ， 坐 标 为 (X-1,7-1) 和 (X, 盖 2)。 注 意 ， 坐 标 (X-1, 盖 1) 一 共 出 现 了 两 次 ; 我 们 稍 
候 再 作 讨论 。 

因此 , 要 找到 一 条 从 原点 出 发 的 路 径 , 我 们 只 需 像 上 面 那样 从 终点 往 回 走 。 从 最 后 一 点 开始 ， 
试 着 找 出 一 条 到 其 相 邻 点 的 路 径 。 下 面 是 该 算法 的 递归 实现 代码 。 








1 public boolean getPath(int x, int y，ArrayList<Point> path) { 
2 Point p = new Point(x, y); 

3 path.add(p); 

4 if (x == 0 && y == 60) { 

5 return true; // 找到 一 条 路 径 

6 } 

也 boolean success = false; 

8 if (x >= 1 && isFree(x - 1，y)) { // 试 着 向 左 

9 success = getPath(x - 1，y，path); // 可 行 ! 向 左 走 

16 

11 if (lsuccess && y >= 1 && isFree(x，y - 1)) { // 试 着 向 上 
12 success = getpath(x, y - 1，path); // 可 行 ! 向 上 走 

13 

14 if (!success) { 

15 path.add(p); // 错 了 ! 最 好 不 要 再 走 这 里 

16 } 

工 7 return success; 

18 } 
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之 前 我 们 提 到 了 重复 路 径 的 问题 。 要 找到 一 条 前 往 (%, 了 的 路 径 ， 就 要 找 出 到 它 的 相 邻 点 





C1, 六 和 (二 1) 的 路 人 径 。 当 然 ， 若 其 中 一 个 方 格 禁 止 通行 ,我们 就 要 绕 着 走 。 接 着 ， 再 看 这 两 
个 点 的 相 邻 点 : (和 2 尺 、( 和 1 天 0)、(C 芝 1 天 0D 和 (有关 2)。 其 中 ，( 芝 1 天 D 出 现 了 两 次 ， 也 意味 
着 我 们 做 了 一 次 无 用 功 。 理 想 情 况 下 , 我 们 应 该 记 下 先前 访问 过 CC1 芒 D ,以免 浪费 宝贵 的 时 间 。 





下 面 就 是 运用 了 动态 规划 的 算法 。 


1 public boolean getpPath(int x, int y，ArrayList<Point> path， 
2 Hashtable<Point, Boolean> cache) { 
3 Point p = new Point(x, y); 

4 if (cache.containsKey(p)) { // 已 访问 过 这 个 点 

5 return cache.get(p); 

6 

7 

8 


} 
path.add(p); 
if (x == 0 && y == 6) { 


9 return true; // 找到 一 条 路 径 

16 } 

半生 boolean success = false; 

12 if (x >= 1 && isFree(x - 1，y)) { // 试 着 向 左 

13 success = getpath(x - 1，y， path，cache); // 可 行 ! 向 左 走 
14 

15 if (lsuccess && y >= 1 && isFree(x, y - 1)) { // 试 着 向 上 
16 success = getpath(x, y - 1，path，cache); // 可 行 ! 向 上 走 
17 

18 if (!success) { 

19 path.add(p); // 错 了 ! 最 好 不 要 再 走 这 里 

26 } 

21 cache.put(p，success); // 缓存 结果 

22 return success; 

23 } 


只 要 稍 作 修 改 ， 就 能 大 幅 提 升 程序 的 执行 速度 。 
9.3 在 数组 A[8...n-1] 中 ， 有 所 谓 的 魔术 索引 ， 满 足 条 件 A[i] = i。 给 定 一 个 有 序 整数 数 





， 元 素 值 各 不 相同 ， 编 写 一 个 方法 ， 在 数组 A 中 找 出 一 个 魔术 索引 ， 若 存在 的 话 。 


进 阶 
如 果 数 组 元 素 有 重复 值 ， 又 该 如 何 处 理 ? 〈 第 68 页 ) 
解法 


看 到 这 个 问题 ， 第 一 反应 可 能 是 选择 蛮 力 法 ,这 也 没什么 好 善 愧 的 。 我 们 可 以 直接 迭代 访问 








整个 数组 ， 找 出 符合 条 件 的 元 素 。 


1 public static int magicSlow(int[] array) { 
2 for (int i = 6; i < array.length; i++) { 
3 if (array[i] == i) { 
4 return i; 
5 } 

6 } 

7 return -1; 
8 


} 
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不 过 ， 既 然 给 定数 组 是 有 序 的 ， 我 们 理应 充分 利用 这 个 条 件 。 

你 可 能 会 发 现 这 个 问题 与 经 典 的 二 分 查找 问题 非常 相似 。 充 分 运用 模式 匹配 法 ,就 能 找 出 适 
当 的 算法 ， 我 们 又 该 怎么 运用 二 分 查找 法 呢 ? 

在 二 分 查找 中 ， 要 找 出 元 素 x， 我 们 会 先 拿 它 跟 数 组 中 间 的 元 素 x 比 较 ， 确 定位 于 x 的 左边 还 
是 右边 。 

以 此 为 基础 ， 是 否 通过 检查 中 间 元 素 就 能 确定 魔术 索引 的 位 置 ? 下 面 来 看 一 个 样 例 数组 : 


























-46 | -26 | -1 1 2 
9 1 2 3 4 


看 到 中 间 元 素 A[5] = 3， 我 们 可 以 断定 魔术 索引 一 定 在 数组 右 侧 ， 因 为 A[mid] < mid。 

为 何 魔术 索引 不 会 在 数组 左 侧 呢 ? 注意 ,从 元 素 ; 赵 至 六 1 时 ， 此 索引 对 应 的 值 至 少 要 减 1, 也 
可 能 更 多 ( 因为 数组 是 有 序 的 ， 且 所 有 元 素 各 不 相同 )。 因 此 ， 如 果 中 间 元 素 就 已 经 太 小 而 不 是 
魔术 索引 的 话 ， 那 么 往 左 侧 移动 时 ， 索 引 减 K， 值 至 少 也 减 K， 所 有 余下 的 元 素 也 会 太 小 。 

继续 运用 这 个 递归 算法 ， 就 会 写 出 与 二 分 查找 非常 相似 的 代码 。 














1 public static int magicFast(int[] array, int start, int end) { 
2 if (end < start || start < 86 || end >= array.length) { 
3 return -1; 
4 
5 int mid = (start + end) / 2; 
6 if (array[mid] == mid) { 
了 return mid; 
} else if (array[mid] > mid){ 
9 return magicFast(array, start, mid - 1); 
16 } else { 
11 return magicFast(array, mid + 1, end); 
12 } 
13 } 
14 


15 public static int magicFast(int[] array) { 
16 return magicFast(array, 0, array.length - 1); 
17 } 


进 阶 :如果 数组 元 素 有 重复 值 又 该 如 何 处 理 ? 
如 果 数 组 元 素 有 重复 值 ， 前 面 的 算法 就 会 失效 。 以 下 面 的 数组 为 例 : 








4|17|9 
6 


eli 


iors | | la 
区 9 


[eel | 3) a 


| 
| 
看 到 A[mid] < mid 时 ， 我 们 无 法 断定 魔术 索引 位 于 数组 哪 一 边 。 它 可 能 在 数组 右 侧 ， 跟 前 
面 一 样 。 或 者 ， 也 可 能 在 左 侧 (在 本 例 中 的 确 在 左 侧 )。 
它 有 没有 可 能 在 左 侧 的 任意 位 置 ? 未 必 。 由 A[5] = 3 可 知 ，A[4] 不 可 能 是 魔术 索引 。A[4] 
必须 等 于 4， 其 索引 才能 成 为 魔术 索引 ， 但 数组 是 有 序 的 ， 故 A[4] 必 定 小 于 A[5]。 


12 | 13 
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事实 上 ， 看 到 A[5] = 3 时 ， 按 照 前 面 的 做 法 ， 我 们 需要 递归 搜索 右 半 部 分 。 不 过 ， 若 搜索 

















左 半 部 分 ， 我 们 可 以 跳 过 一 些 元 素 ， 只 递归 搜索 A[6] 到 A[3] 的 元 素 。A[3] 是 第 一 个 可 能 成 为 魔 


术 索 引 的 元 素 。 


综 上 ， 我 们 得 到 一 般 模式 : 先 比较 midIndex 和 midvalue 是 否 相 同 。 然 后 ， 大 两 者 不 同 ， 则 


按 如 下 方式 递归 搜索 左 半 部 分 和 右 半 部 分 。 





口 右 半 部 分 : 搜索 索引 从 Math.max(midIndex + 1，midvalue) 到 end 的 元 素 。 
下 面 是 该 算法 的 实现 代码 。 





1 public static int magicFast(int[] array, int start, int end) { 
2 if (end < start || start < 86 || end >= array.length) { 

3 return -1; 

4 

5 int midIndex = (start + end) / 2; 

6 int midValue = array[midIndex]; 

7 if (midValue == midIndex) { 

8 return midIndex; 

9 } 

106 


11 /* 搜索 左 半 部 分 */ 

12 int leftIndex = Math.min(midIndex - 1, midValue); 
13 int left = magicFast(array, start, leftIndex); 

14 if (left >= 6) { 

15 return left; 

16 } 


18 /* 搜索 右 半 部 分 */ 
19 int rightIndex = Math.max(midIndex + 1, midValue); 
26 int right = magicFast(array，rightIndex，end); 


22 return right; 
23 } 


25 public static int magicFast(int[] array) { 
26 return magicFast(array, 86, array.length - 1); 
27 } 








口 左 半 部 分 : 搜索 索引 从 start 到 Math.min(midIndex - 1，midValue) 的 元 素 。 


注意 , 在 上 面 的 代码 中 , 如 果 数 组 元 素 各 不 相同 , 这 个 方法 的 执行 动作 与 第 一 个 解法 几 近 相同 。 


9.4 ”编写 一 个 方法 ， 返 回 某 集合 的 所 有 子 集 。( 第 68 页 ) 
解法 








着 手 解决 这 个 问题 之 前 , 我 们 先 要 对 时 间 和 空间 复杂 度 有 个 合理 的 评估 。 一 个 集合 会 有 多 少 
子 集 ? 我 们 可 以 这 么 计算 ， 生 成 一 个 子 集 时 ， 每 个 元 素 都 可 以 “选择 ”在 或 不 在 这 个 子 集中 。 也 




















就 是 说 ， 第 一 个 元 素 有 两 个 选择 : 它 要 么 在 集合 中 ,要 么 不 在 集合 中 。 同 样 ， 第 二 个 元 素 也 有 两 
个 选择 ， 依 此 类 推 ，2 相 乘 z 次 ，{2 * 2 * … } 等 于 2 个 子 集 。 因 此 ， 在 时 间或 空间 复杂 度 上 ， 





我 们 不 可 能 做 得 比 0(2”) 更 好 。 
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集合 {qi, az …,a 直 的 所 有 子 集 组 成 的 集合 也 称 为 寡 集 (powerset ), 用 符号 表示 为 :P({al az …， 


an}) 或 P(n)。 
解法 1: 递归 


此 题 非常 适合 采用 简单 构造 法 。 假 设 我 们 正 尝试 找 出 集合 8 = {a qz, … aw} 的 所 有 子 集 ， 可 


从 终止 条 件 开始 。 
@ 终止 条 件 : n=0 
空 集合 只 有 一 个 子 集 : {}。 
@ 情况 : n= 1 
集合 {al} 有 两 个 子 集 : {}、{ai}。 
@ 情况 : n=2 
集合 {ai, ao} 有 四 个 子 集 : 所 {ai} 、 {a2} 、 {a1, a2}o 


@ 情况 : n=3 





至 此 , 事情 开始 变 得 有 点 意思 了 。 我 们 想 找 出 一 种 方法 ,可 以 根据 之 前 的 答案 产生 n=3 时 的 


答案 


也 


O 





n ==3 和 n =2 的 两 个 答案 之 间 有 何不 同 ? 下 面 让 我 们 更 细致 地 分 析 两 者 差异 : 


PO)= 人 {a1}, {a2}, {a1, a2} 
P(3) 人 {a1}, {a2}, {a3}, {a1, 02}， {an a3}, {a2, a3}, 


{a1, az a3} 


两 者 之 间 的 不 同 之 处 在 于 ， 所 有 含有 as 的 子 集 ，P(2) 都 没有 。 


PG) - PQ)= {a3}, {a1, a3}, {42, a3}, {a1, a2, a3} 
那么 ， 我 们 该 如 何 利用 P(2) 构 造 P(3)? 很 简单 ， 只 需 复 4 
加 a;: 
PQ2)= {0}, {a1}, {4a2}, {a1, 42} 
P(2) + a3= {a3}, {a1, a3}, {a a3}, {a1, 42, a3} 
两 者 合并 在 一 起 ， 即 可 产生 P(3)。 


@ 情况 : n>0 





制 P(2) 里 的 子 





， 并 在 这 些 子 集中 添 


只 要 将 上 述 步 骤 稍 作 一 般 化 处 理 , 就 能 产生 一 般 情况 的 P(n): 先 计 算 P(n-1), 复制 一 份 结果 ， 


然后 在 每 个 复制 后 的 集合 中 加 入 an。 
下 面 是 该 算法 的 实现 代码 : 


























ArrayList<ArrayList<Integer>> getSubsets(ArrayList<Integer> set， 


int index) 
ArrayList<ArrayList<Integer>> allsubsets; 
if (set.size() == index) { // 终止 条 件 ， 加 入 空 集合 


{ 


allsubsets = new ArrayList<ArrayList<Integer>>(); 


= 


else { 

allsubsets = getSubsets(set, index + 1); 
int item = set.get(index); 

0 ArrayList<ArrayList<Integer>> moresubsets = 
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1 
2 
3 
4 
5 
6 allsubsets.add(new ArrayList<Integer>()); // 空 集合 
了 
8 
9 
1 








11 new ArrayList<ArrayList<Integer>>(); 

12 for (ArrayList<Integer> subset : allsubsets) { 
13 ArrayList<Integer> newsubset = new ArrayList<Integer>(); 
14 newsubset.addAll(subset); 

15 newsubset .add(item); 

16 moresubsets.add(newsubset); 

17 } 

18 allsubsets.addAll (moresubsets); 

19 } 

26 return allsubsets; 

21 } 

















这 个 解法 的 时 间 和 空间 复杂 度 为 02)， 已 经 是 最 优 解 。 非 要 锦上添花 的 话 ， 我 们 还 


代 方 式 实现 这 个 算法 。 


解法 2: 组 合 数 学 (Combinatorics) 

















尽管 上 面 的 解法 没什么 地 方 不 对 ， 不 过 还 是 可 以 男 疯 他 法 ,解决 这 个 问题 。 
回想 一 下 ， 在 构造 一 个 集合 时 ， 每 个 元 素 有 两 个 选择 ; ( 1 ) 该 元 素 在 这 个 集合 中 (“yes” 状 


态 ), 或 者 (2 ) 该 元 素 不 在 这 个 集合 中 (“no” 状 态 )。 


比如 “yes， ye no, no, yes, no ”。 














这 就 意味 着 每 个 子 集 都 是 一 串 yes 和 no， 


由 此 ， 总 共 可 能 会 有 2 个 子 集 。 怎 样 才能 迭代 遍历 所 有 元 素 的 所 有 “yes”/“no” 序 列 ? 如 





果 将 每 个 “yes” 视 作 1， 每 个 “no” 视 作 0， 那 么 ， 每 个 子 集 就 可 以 表示 为 一 个 二 进 制 串 。 











接着 ， 构 和 告 所 有 子 集 就 等 同 于 构造 所 有 的 二 进 制 数 ( 也 即 所 有 整数 )。 我 们 会 迭代 访问 1 到 2” 





的 所 有 数字 ， 再 将 这 些 数字 的 二 进 制 表示 转换 成 集合 。 


小 事 一 桩 ! 


ArrayList<ArrayList<Integer>> getSubsets2(ArrayList<Integer> set) { 


ArrayList<ArrayList<Integer>> allsubsets = 
new ArrayList<ArrayList<Integer>>(); 
int max = 1 << set.size(); /* 计算 2^n */ 


ArrayList<Integer> subset = convertIntToSet(k, set); 


allsubsets.add(subset); 


} 


1 
之 
3 
4 
5 for (int k = 8@; k < max; k++) { 
6 
yi 
8 
9 return allsubsets; 


16 } 


12 ArrayList<Integer> convertIntToSet(int x, ArrayList<Integer> set) { 
13 ArrayList<Integer> subset = new ArrayList<Integer>(); 


14 int index = 0@; 
15 for (int k = x; k > 6; k >>= 1) { 


16 if ((k & 1) == 1) { 

17 subset.add(set.get(index)); 
18 } 

19 index++; 

26 } 

2 下 return subset; 

22 } 


相 比 前 一 种 解法 ， 这 种 解法 不 存在 实质 的 差异 ， 并 无 上 下 之 分 。 
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9.5 ”编写 一 个 方法 ， 确 定 某 字符 串 的 所 有 排列 组 合 。( 第 68 页 ) 

解法 

跟 许多 递归 问题 一 样 ， 简 单 构造 法 非常 管用 。 假 设 有 个 字符 串 S， 以 字符 序列 wa>…aw 表 示 。 

@ 终止 条 件 : n=1 

S=a， 只 有 一 种 排列 组 合 ， 即 字符 串 wi。 

@ 情况 : n=2 

S= ai， 有 两 种 排列 组 合 ww 和 azai。 

@ 情况 : n=3 

至 此 , 情况 变 得 越 来 越 有 意思 。 根 据 aiaz 的 排列 组 合 , 如 何 产生 aiazas 的 所 有 排列 组 合 呢 ? 也 
就 是 说 ， 给 定 





























U142, 4241 
我 们 需要 产生 : 
U10203, A10302, 4d24143, G24341, 03C102，030201 
这 两 个 字符 序列 的 区 别 在 于 前 者 不 含 a;, 而 后 者 包含 a;。 那么 , 怎样 才能 根据 X2) 生 成 X3) 呢 ? 
很 简单 ， 将 a; 塞 进 R2) 里 所 有 字符 串 的 任意 可 能 位 置 即 可 。 
@ 情况 : n>0 


对 于 一 般 情 况 ， 我 们 只 需 重复 这 个 步骤 。 既 然 已 求 得 Fo-1) 的 解 ， 接 着 只 要 将 w 搬 入 这 些 字 














符 串 的 任意 位 置 。 
具体 代码 如 下 。 
1 public static ArrayList<String> getPerms(String str) { 
2 if (str == null) { 
3 return null; 
4 } 
5 ArrayList<String> permutations = new ArrayList<String>(); 
6 if (str.length() == 6) { // 终止 条 件 
7 permutations.add(™”); 
8 return permutations; 
9 } 
16 


11 char first = str.charAt(6); // 取得 第 一 个 字符 
12 String remainder = str.substring(1); // 移 除 第 一 个 字符 


13 ArrayList<String> words = getPerms(remainder); 
14 for (String word : words) { 

15 for (int j = 6; j <= word.length(); j++) { 
16 String s = insertCharAt(word, first, j); 
17 permutations.add(s); 

18 } 

19 } 

26 return permutations; 

pp 

22 


23 public static String insertCharAt(String word, char c, int i) { 
24 String start = word.substring(60, i); 
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25 String end = word.substring(i); 
26 return start + c + end; 
P70 


由 于 将 会 有 nl! 种 排列 组 合 ， 这 种 解法 的 时 间 复 杂 度 为 O(n!)， 已 经 是 最 优 解 。 
9.6 “实现 一 种 算法 ， 打 印 " 对 括号 的 全 部 有 效 组 合 〈 即 左右 括号 正确 配对 )。( 第 68 页) 


解法 
看 到 此 题 ， 我 们 的 第 一 反应 可 能 是 运用 递归 法 ,将 一 对 括号 加 进 fn-1) 的 解答 ， 从 而 得 到 fn) 
的 解答 。 从 直觉 上 看 ， 这 个 方法 不 错 。 
下 面 来 看 看 n = 3 时 的 答案 : 
(()()) ((())) ()(()) (())() ()()() 
如 何以 n = 2 时 的 答案 为 基础 构建 上 面 的 结果 呢 ? 
(()) ()() 
我 们 可 以 在 字符 串 最 前 面 以 及 原 有 的 每 对 括号 里 面 插入 一 对 括号 。 至 于 插 和 人 其 他 任意 位 置 ， 
比如 字符 串 的 末尾 ， 都 会 跟 之 前 的 情况 重复 。 
综 上 ， 可 得 到 以 下 结 
(()) ->(()()) /* 在 第 1 个 堪 括 号 之 后 插入 一 对 括号 */ 
-> ((())) /* 在 第 2 个 左 括号 之 后 插入 一 对 括号 */ 
-> ()(()) /* 在 字符 串 开 头 插入 一 对 括号 */ 
()() -> (())() /* 在 第 1 个 左 括号 之 后 插入 一 对 括号 */ 


-> ()(()) /* 在 第 2 个 左 括号 之 后 插入 一 对 括号 */ 
-> ()()() /* 在 字符 串 开 头 播 入 一 对 括号 */ 


且慢 ， 上 面 有 重复 的 括号 对 组 合 ，()(() ) 出 现 了 两 次 。 
如 果 准 备 采 用 这 种 做 法 ,那么 ， 将 字符 串 放 进 结果 列表 之 前 ， 必 须 先 检查 有 无 重复 。 



































1 public static Set<String> generateparens(int remaining) { 
2 Set<String> set = new HashSet<String>(); 

3 if (remaining == 6) { 

4 set.add(”); 

5 } else { 

6 Set<String> prev = generateparens(remaining - 1); 
4 for (String str : prev) { 

8 for (int i = 60; i < str.length(); i++) { 

9 if (str.charAt(i) == ‘(‘) { 

16 String s = insertInside(str, i); 

11 /* 若 s 不 在 Set 中 ， 则 将 s 插 入 set 中 。 注 意 ， 

12 * 插入 元 素 之 前 ，HashSet 会 自动 检查 有 无 重复 ， 
13 * 因此 没 必 要 专门 检查 元 素 是 否 重 复 */ 

14 set.add(s); 

15 } 

16 

17 if (!set.contains(“()” + str)) { 

18 set.add(“()” + str); 

19 } 

26 } 
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21 } 

22 return set; 
23 } 

24 


25 public String insertInside(String str, int leftIndex) { 
26 String left = str.substring(8, leftIndex + 1); 
27 String right = str.substring(leftIndex + 1, str.length()); 


28 return left + “()” + right; 





这 种 做 法 可 行 ， 但 效率 不 太 高 ， 在 排查 重 








复 字符 串 上 浪费 了 大 量 时间 。 





男 一 种 解法 是 从 头 开始 构造 字符 串 ， 从 而 避免 出 现 重 复 字 符 串 。 在 这 个 解法 中 , 逐一 加 入 左 
括号 和 右 括号 ， 只 要 字符 串 仍 然 有 效 ( 合乎 题 意 )。 





每 次 递归 调用 , 都 会 有 个 索引 值 向 字符 
何 时 可 以 用 左 插 号 ， 何 时 可 以 用 右 括号 呢 ? 








的 茶 个 


字符 。 我 们 需要 选择 左 括号 或 右 括号 , 那么 ， 


(1) 左 括号 :; 只 要 左 括号 还 没有 用 完 ， 就 可 以 插入 左 括号 。 
(2) 右 括号 : 只 要 不 造成 语法 错误 ， 就 可 以 搬入 右 括号 。 何 时 会 出 现 语法 错误 ? 如 果 右 括号 


比 左 括号 还 多 ， 就 会 出 现 语法 错误 。 


因此 , 我 们 只 需 记 录 人 允许 插入 的 左右 括号 数目 。 如 果 还 有 左 括号 可 用 ,就 插入 一 个 左 括号 然 
后 递归 。 如 果 右 括号 比 左 括号 还 多 ( 也 就 是 使 用 中 的 左 括号 比 右 括号 还 多 )， 就 插入 一 个 右 括号 





然后 递归 。 
1 public void addParen(ArrayList<Sstring> list, int leftRem, 
2 int rightRem, char[] str, int count) { 
3 if (leftRem < 8 || rightRem < leftRem) return; // 无 效 状态 
4 
5 if (leftRem == 6 && rightRem == 6) { /* 没有 括号 可 用 了 */ 
6 String s = String.copyValueOf(str); 
yd list.add(s); 
8 } else { 
9 /* 若 还 有 左 括 号 可 用 ， 则 加 入 一 个 左 括号 */ 
16 if (leftRem > 6) { 
了 str[count] = “(’?; 
12 addParen(list, leftRem - 1, rightRem, str, count + 1); 
13 } 
14 
15 /* 若 字 符 囊 是 有 效 的 ， 则 加 入 右 括 号 */ 
16 if (rightRem > leftRem) { 
17 str[count] = )’; 
18 addParen(list, leftRem, rightRem - 1, str, count + 1); 
19 } 
26 } 
21 } 
22 
23 public ArrayList<String> generatepParens(int count) { 
24 char[] str = new char[count*2]; 
25 ArrayList<String> list = new ArrayList<String>(); 
26 addParen(list, count, count, str, 6); 
27 return list; 
28 } 


图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 





232 第 9 章 解 题 技 巧 











因为 我 们 是 在 字符 串 的 每 一 个 索引 对 应 位 置 搬入 左 括号 和 右 括号 ,而 且 绝 不 会 重复 索引 , 所 
以 ， 可 以 保证 每 个 字符 串 都 是 独一无二 的 。 


9.7 ”编写 函数 ， 实 现 许 多 图 片 编辑 软件 都 支持 的 “填充 颜色 ”功能 。 给 定 一 个 屏幕 〈 以 二 
维 数组 表示 ， 元 素 为 颜色 值 )、 一 个 点 和 一 个 新 的 颜色 值 ， 将 新 颜色 值 填 入 这 个 点 的 周围 区 域 ， 
直到 原来 的 颜色 值 全 都 改变 。( 第 69 页) 


解法 

首先 ， 想 象 一 下 这 个 方法 是 怎么 回 事 。 假 设 要 对 一 个 像素 ( 比如 绿色 ) 调用 paintFil1 (也 
即 点 击 图 片 编辑 软件 的 填充 颜色 ), 我 们 希望 颜色 向 四 周 “ 渗 出 ”。 我们 会 对 周围 的 像素 逐一 调用 
paintFill1， 向 外 扩张 ， 一旦 磁 到 非 绿 色 的 像素 就 停止 填充 。 

我 们 可 以 递归 方式 实现 这 个 算法 : 

1 enum Color { 


Black, White, Red, Yellow, Green 


} 


2 
3 
4 
5 boolean paintFill(Color[][] screen, int x, int y, Color ocolor, 
6 
7 
8 
























































Ill 





Color ncolor) { 
if (x < 0 || x >= screen[6].length || 
y < 0 ||y >= screen.length) { 


9 return false; 

16 } 

11 if (screen[y][x] == ocolor) { 

12 screen[y][x] = ncolor; 

13 paintFill(screen, x - 1, y, ocolor, ncolor); // 左 
14 paintFill(screen, x + 1, y, ocolor, ncolor); // 右 
15 paintFill(screen, x, y - 1, ocolor, ncolor); // 上 
16 paintFill(screen, x, y + 1, ocolor, ncolor); // 下 
17 } 

18 return true; 

19 } 

26 

21 boolean paintFill(Color[][] screen, int x, int y, Color ncolor) { 
22 if (screen[y][x] == ncolor) return false; 

23 return paintFill(screen, x, y, screen[y][x], ncolor); 
24 } 


注意 screen[y][x] 中 x 和 y 的 顺序 ， 碰 到 图 像 问题 时 切记 这 一 点 。 因 为 x 表示 水 平 轴 (也 即 自 左 
向 右 )， 实际 上 对 应 于 列 数 而 非 行 数 。y 的 值 等 于 行 数 。 在 面试 以 及 平时 写 代码 时 ， 这 个 地 方 也 很 
容 犯 错 。 

9.8 给 定数 量 不 限 的 硬币 ， 币 值 为 25 分 、10 分 、5 分 和 1 分 ， 编 写 代 码 计算 "分 有 几 
种 表示 法 。( 第 69 页 ) 

解法 

这 是 个 递归 问题 ， 我 们 要 找 出 如 何 利 用 较 早 的 答案 (也 就 是 子 问题 的 答案 ) 计算 出 
makeChange(n)。 
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假设 n = 100， 我 们 想 要 算出 100 分 有 几 种 换 零 方式 。 这 个 问题 与 其 子 问 题 之 间 有 何 关系 呢 ? 
我 们 知道 100 分 换 零 后 会 包含 0、1、2 、3 或 4 个 25 分 硬币 (quarter )， 因 此 : 


makeChange(166) = 
makeChange(168， 使 用 8 个 25 分 硬币 ) 
makeChange(168， 使 用 1 个 25 分 硬币 ) 
makeChange(169， 使 用 2 个 25 分 硬币 ) 
makeChange(169， 使 用 3 个 25 分 硬币 ) 
makeChange(168， 使 用 4 个 25 分 硬币 ) 


仔细 观察 一 番 ， 可 以 看 出 其 中 有 些 问题 简化 掉 了 。 举 个 例子 ，makechange(188， 使 用 1 个 25 
分 硬币 ) 与 nakeChange(75， 使 用 6 个 25 分 硬币 ) 等 价 。 这 是 因为 ， 如 果 给 100 分 换 零 时 只 准 用 1 个 25 
分 硬币 ,那么 ， 我 们 就 只 能 选择 给 余下 的 75 分 换 零 。 

同样 的 逻辑 也 适用 于 makeChange(168， 使 用 2 个 25 分 人 硬币) 、makeChange(1686， 使 用 3 个 25 分 
硬币 ) 和 makechange(168， 使 用 4 个 25 分 硬币 ) 。 综 上 ， 前 面 的 算式 可 简化 为 : 


makeChange(166) = 
makeChange(166， 使 用 8 个 25 分 硬币 ) + 
makeChange(75， 使 用 9 个 25 分 硬币 ) + 
makeChange(58， 使 用 9 个 25 分 硬币 ) + 
makeChange(25， 使 用 9 个 25 分 硬币 ) + 
1 


注意 最 后 一 行 ，makeChange(168， 使 用 4 个 25 分 硬币 ) 等 于 1。 我 们 把 这 叫 作 “ 完 全 简化 ”。 

接 下 来 呢 ? 我 们 已 经 用 完了 25 分 硬币 ， 现 在 可 以 开始 使 用 下 一 个 币值 最 大 的 硬币 : 10 分 硬 
币 ( dime )g 

前 面 使 用 25 分 硬币 的 做 法 同样 可 以 套用 在 10 分 硬币 上 , 但 需要 套用 在 上 面 算式 五 部 分 中 的 四 
个 部 分 ， 且 每 一 部 分 都 要 套用 。 第 一 部 分 的 套用 结果 如 下 : 


makeChange(1686， 使 用 6 个 25 分 硬币 ) = 
makeChange(166， 使 用 8 个 25 分 硬币 、6 个 1 分 硬币 ) + 
makeChange(166， 使 用 6 个 25 分 硬币 、1 个 16 分 硬币 ) + 
makeChange(168， 使 用 @ 个 25 分 硬币 、2 个 16 分 硬币 ) + 





+ 十 二 十 























makeChange(1868， 使 用 6 个 25 分 硬币 、16 个 16 分 硬币 ) 


makeChange(75， 使 用 8 个 25 分 硬币 ) = 
makeChange(75， 使 用 6 个 25 分 硬币 、6 个 18 分 硬币 ) + 
makeChange(75， 使 用 8 个 25 分 硬币 、1 个 16 分 硬币 ) + 
makeChange(75， 使 用 8 个 25 分 硬币 、2 个 16 分 硬币 ) + 


makeChange(75， 使 用 6 个 25 分 硬币 、7 个 16 分 硬币 ) 


makeChange(58， 使 用 8 个 25 分 硬币 ) = 
makeChange(58， 使 用 8 个 25 分 硬币 、8 个 18 分 硬币 ) + 
makeChange(58， 使 用 8 个 25 分 硬币 、1 个 18 分 硬币 ) + 
makeChange(58， 使 用 8 个 25 分 硬币 、2 个 16 分 硬币 ) + 








makeChange(58， 使 用 6 个 25 分 硬币 、5 个 18 分 硬币 ) 
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makeChange(25， 使 用 8 个 25 分 硬币 ) = 
makeChange(25， 使 用 6 个 25 分 硬币 、6 个 18 分 硬币 ) + 
makeChange(25， 使 用 8 个 25 分 硬币 、1 个 18 分 硬币 ) + 
makeChange(25， 使 用 9 个 25 分 硬币 、2 个 18 分 硬币 ) 


开始 使 用 5 分 镍 币 (mickel ) 时 ， 上 面 算式 的 每 一 部 分 都 要 逐一 展开 ， 最 终 会 得 到 一 个 树 状 递 
归结 构 ， 其 中 每 次 调用 都 会 展开 为 4 个 或 更 多 调用 。 

















递归 的 终止 条 件 就 是 完全 简化 的 算式 。 举 个 例子 ，makechange(58， 使 用 8 个 25 分 硬币 、5 个 
18 分 硬币 ) 会 被 完全 简化 为 1， 因 为 5 个 10 分 硬币 就 等 于 50 分 。 
由 上 述说 明 可 导出 类 似 如 下 的 递归 算法 : 





1 public int makeChange(int n, int denom) { 
之 int next denom = 0@; 
3 switch (denom) { 

4 Case 25: 

5 next_denom = 108; 
6 break; 

7 case 16: 

8 next_denom = 5; 

9 break ; 

16 case 5: 

11 next_denom = 1; 
12 break; 

13 case 1: 

14 return 1; 

15 } 

16 


17 int ways = 
18 for (int i 


8; 
= 0; i * denom <= nj i++) { 


19 ways += makeChange(n - i * denom, next_denom); 
26 } 

2 return ways; 

22 } 

和 23 


24 System.out.writeln(makeChange(160, 25)); 


上 面 的 算法 只 适用 于 美国 币值 ， 不 过 ， 只 要 稍 加 修改 扩充 ， 就 能 用 于 其 他 的 币值 组 合 。 


9.9 设计 一 种 算法 ， 打 印 八 皇后 在 8x8 棋盘 上 的 各 种 摆 法 ， 其 中 每 个 皇后 都 不 同行 、 不 同 
列 ， 也 不 在 对 角 线 上 。 这 里 的 “对 角 线 ” 指 的 是 所 有 的 对 角 线 ， 不 只 是 平分 整个 棋盘 的 那 两 条 对 


角 线 。( 第 69 页 ) 
解法 














我 们 必须 在 8x8 棋 

















盘 上 排 好 8 个 皇后 , 每 个 皇后 都 位 于 不 同行 、 不 同 列 , 且 不 在 同一 对 角 线 上 。 








由 此 可 知 ， 每 一 行 、 每 一 列 以 及 对 角 线 只 能 使 用 一 次 。 
想象 一 下 最 后 放 到 棋盘 上 的 那个 皇后 , 这 里 假设 是 在 第 8 行 。( 这 么 假设 没有 问题 , 因为 这 些 
皇后 怎么 摆 放 都 没关系 。) 这 个 皇后 要 摆 在 第 8 行 的 哪 一 格 呢 ? 一 共有 8 种 选择 ， 每 一 列 代表 一 种 





习 


台 已 
HEo 
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“摆好 ” 八 皇 后 的 棋盘 ， 其 中 一 种 摆 法 


因此 ， 欲 知 八 皇后 在 8x8 棋 盘 上 的 所 有 可 能 摆 法 ， 具 体 算法 如 下 : 


八 皇 后 在 8x8 棋 盘 上 的 摆 法 = 
八 皇 后 在 8x8 棋 盘 上 的 摆 法 ， 且 其 中 一 个 皇后 位 于 (7，8) 
八 皇 后 在 8x8 棋 盘 上 的 摆 法 ， 1 1) 
八 皇 后 在 8x8 棋 盘 上 的 摆 法 ， 且 其 中 一 个 皇后 位 于 (7，2) 
八 皇 后 在 8x8 棋 盘 上 的 摆 法 ， FP 后 位 于 (7，3) 
八 皇 后 在 8x8 棋 盘 上 的 摆 法 ， 且 其 中 一 个 皇后 位 于 (7，4) 
八 皇 后 在 8x8 棋 盘 上 的 摆 法 ， 且 其 中 一 个 皇后 位 于 (7，5) 
八 皇 后 在 8x8 棋 盘 上 的 摆 法 ， ee 6) 
八 皇后 在 8x8 棋 盘 上 的 摆 法 ， 且 其 中 一 个 皇后 位 于 (7，7) 


接着 ， 运 用 非常 类 似 的 方法 计算 其 中 的 每 项 : 


八 皇 后 在 8x8 棋 盘 上 的 摆 法 ， 且 其 中 一 个 皇后 位 于 (7，3) = 
八 皇后 在 .…. 的 摆 法 ， 且 其 中 两 个 皇 3) 和 (6，6) 
八 皇 后 在 …. 的 摆 法 ， 且 其 中 两 个 皇后 位 于 (7，3) 和 (6，1) 
八 皇 后 在 .…. 的 摆 法 ， 且 其 中 两 个 皇后 位 于 (7，3) 和 (6，2) 
八 皇 后 在 .… 的 摆 法 ， 且 其 中 两 个 皇后 位 于 (7，3) 和 (6，4) 
八 皇 后 在 .…. 的 摆 法 ， 且 其 中 两 个 皇后 位 于 (7，3) 和 (6，5) 
八 皇 后 在 …. 的 摆 法 ， 且 其 中 两 个 皇后 位 于 (7，3) 和 (6，6) 
八 皇 后 在 …. 的 摆 法 ， 且 其 中 两 个 皇后 位 于 (7，3) 和 (6，7) 


意 ， 我 们 不 必 考 虑 皇后 位 于 格子 (7，3) 和 (6，3) 的 组 合 情 况 ， 因 为 这 与 所 有 皇后 不 同行 、 
不 同 列 且 不 在 对 角 线 上 的 要 求 不 符 。 
接 下 来 ， 具 体 实 现 也 就 相当 简单 了 。 

















+ 二 十 二 ++ 十 十 








+ 十 十 十 二 ++ 








int GRID_SIZE = 8; 


void placeQueens(int row, Integer[] columns, 
ArrayList<Integer[]> results) { 


1 
2 
3 
4 
5 if (row == GRID_SIZE) { // 找到 有 效 的 摆 法 
6 results.add(columns.clone()); 

2 

8 





} else { 
for (int col = 6; col < GRID_ SIZE; col++) { 

9 if (checkValid(columns, row, col)) { 
16 columns[row] = col; // 摆 放 皇后 
11 placeQueens(row + 1, columns, results); 
12 } 
13 } 
14 } 
15 } 
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17 /* 检查 (Frow1，column1) 可 和 否 摆 放 皇后 ， 做 法 是 
18 * 检查 有 无 其 他 皇后 位 于 同一 列 或 对 角 线 ， 不 必 
19 * 检查 是 否 在 同一 行 上 ， 因 为 调用 placeQueen 时 ， 
28 * 一 次 只 会 摆 放 一 个 皇后 ， 由 此 可 知 ， 这 一 行 是 


21 * 空 的 */ 

22 boolean checkValid(Integer[] columns, int Pow1，int column1) { 
23 for (int row2 = 6j row2 < rowl; row2++) { 

24 int column2 = columns[row2]; 

25 /* 检查 (Fow2，Ccolumn2) 是 否 会 让 (row1，column1) 变 成 无 效 
26 * 摆 放 位 置 */ 

27 

28 /* 检查 同一 列 有 无 其 他 皇后 */ 

29 if (column1 == column2) { 

36 Peturn false; 

31 } 

32 

33 /* 检查 对 角 线 : 若 两 列 的 距离 等 于 

34 * 两 行 的 距离 ， 就 表示 两 个 皇后 在 

35 * 同一 对 角 线 上 */ 

36 int columnDistance = Math.abs(column2 - column1); 
37 

38 /* rowl1 > row2,， 不 用 取 绝 对 值 */ 

39 int rowDistance = row1l - row2; 

46 if (columnDistance == rowDistance) { 

41 return false; 

42 } 

43 } 

44 return true; 

45 } 





注意 , 每 一 行 只 能 摆 放 一 个 皇后 ,因此 不 需要 将 棋盘 储存 为 完整 的 8x8 和 矩阵 ， 只 需 一 维 数组 ， 
其 中 columns[r] = c 表 示 有 个 皇后 位 于 行 r 列 c。 


9.10 给 你 一 堆 n 个 箱子 ， 箱 子 宽 w;、 高 h;/、 深 di;。 箱 子 不 能 翻转 ， 将 箱子 堆 起 来 时 ， 下 面 
箱子 的 宽度 、 高 度 和 深度 必须 大 于 上 面 的 箱子 。 实 现 一 个 方法 ， 搭 出 最 高 的 一 堆 箱子 ， 箱 堆 的 
高 度 为 每 个 箱子 高 度 的 总 和 。( 第 69 页 ) 


解法 

要 解决 此 题 ， 我 们 需要 找到 不 同 子 问题 之 间 的 关系 。 

假设 我 们 有 以 下 这 些 箱 子 : b1, b,,…, b,。 能 够 堆 出 的 最 高 箱 堆 的 高 度 等 于 max( 底 部 为 51 的 最 
高 箱 堆 , 底部 为 5 的 最 高 箱 堆 , …, 底部 为 5, 的 最 高 箱 堆 )。 也 就 是 说 ， 只 要 试 着 用 每 个 箱子 作为 箱 
堆 底部 并 搭 出 可 能 的 最 高 高 度 ， 就 能 找 出 箱 堆 的 最 高 高 度 。 

但 是 , 该 怎么 找 出 以 某 个 箱子 为 底 的 最 高 箱 堆 呢 ? 具体 做 法 与 之 前 的 完全 相同 。 我 们 会 试 着 
在 第 二 层 以 不 同 的 箱子 为 底 继 续 堆 箱子 ， 如 此 反复 。 

当然 ,我们 只 需 尝 试 有 效 的 箱子 ， 也 就 是 说 ， 若 bs 大 于 b1， 那 就 不 必 尝 试 这 么 堆 箱 子 : {1， 
bs,…}， 因 为 bp 不 能 放 在 bs 下 面 。 
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下 面 是 该 算法 的 递归 实现 代码 。 


1 public ArrayList<Box> createStackR(Box[] boxes, Box bottom) { 

之 int max_height = 6 

3 ArrayList<Box> max_stack = null; 

4 for (int i = 6; i < boxes.length; i++) { 

5 if (boxes[i].canBeAbove(bottom)) { 

6 ArrayList<Box> new_stack = createStackR(boxes, boxes[i]); 
7 int new height = stackHeight(new_stack); 

8 if (new_height > max_height) { 

9 max_stack = new_stack; 

16 max_height = new_height; 


12 } 
13  } 


15 if (max_stack == null) { 

16 max_stack = new ArrayList<Box>(); 

jy } 

18 if (bottom != null) { 

19 max_stack.add(8，bottom); // 插入 箱 堆 底部 
20 } 


22 return max_stack; 

23 } 

上 述 代 码 的 问题 是 效率 太 低 , 我 们 可 能 已 经 找 出 以 bs 为 底 A 但 还 是 尝试 找到 类 似 {b，， 

…} 的 最 佳 解决 方案 。 我 们 不 必 像 之 前 那样 从 零 开 始 构 造 这 些 管 案 ， 完 全 可 以 运用 动态 规划 ， 
Ss 


1 public ArrayList<Box> createStackDP(Box[] boxes, Box bottom, 
2 HashMap<Box, ArrayList<Box>> stack_map) { 

















3 if (bottom != null && stack map.containsKey(bottom)) { 
4 return stack_map.get(bottom); 

5 } 

6 

7 int max_height = ©@; 

8 ArrayList<Box> max_stack = null; 

9 for (int i = 6; i < boxes.length; i++) { 

16 if (boxes[i].canBeAbove(bottom)) { 

11 ArrayList<Box> new_stack = 

12 createSstackDP(boxes, boxes[i], stack_map); 

13 int new height = stackHeight(new_stack); 

14 if (new_ height > max_height) { 

15 max_stack = new_stack; 

16 max_height = new_height; 

17 } 

18 } 

19 } 

20 

21 if (max_stack == null) max_stack = new ArrayList<Box>(); 


22 if (bottom != null) max_stack.add(86，bottom) ; 
23 stack_map.put(bottom, max_stack); 
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25 return (ArrayList<Box>)max_stack.clone(); 
26 } 


你 可 能 会 问 ， 第 25 行 代码 为 什么 非 要 转型 max_stack.clone()，max_stack 不 就 已 经 是 正确 
的 数据 类 型 了 吗 ? 没 错 ， 但 我 们 还 是 需要 进行 转型 。 

方法 clone() 来 自 0bject 类 ， 其 方法 签名 如 下 : 

1 protected Object clone() { ... } 
重 写 方 法 时 ， 可 以 调整 参数 ， 但 不 得 改动 返回 类 型 。 因 此 ， 如 果 继 承 自 object 的 Foo 类 重 写 
了 clone()， 它 的 clone() 方 法 仍 将 返回 Object 实例 。 

这 正 是 语句 (ArrayList<Box>)max_stack.clone() 的 真正 作用 。 这 个 类 会 重 写 clone(), 但 
该 方法 仍然 会 返回 object， 因 此 ， 我 们 必须 转型 返回 值 。 


9.11 给 定 一 个 布尔 表达 式 ， 由 6、1、&\、| 和 “等 符号 组 成 ， 以 及 一 个 想 要 的 布尔 结果 
result， 实 现 一 个 图 数 ， 算 出 有 几 种 括号 的 放 法 可 使 该 表达 式 得 出 result 值 。( 第 69 页 ) 


解法 
跟 其 他 递归 问题 一 样 ， 此 题 的 关键 在 于 找 出 问题 与 子 问题 之 间 的 关系 。 
假设 函数 int f(expression，result) 会 返回 所 有 值 为 return 的 有 效 表 达 式 的 数量 。 我 们 
想 要 算出 f(1^*8|6|1，true) (也 即 ， 给 表达 式 1^61811 加 括号 使 其 求 值 为 true 的 所 有 方式 )。 每 
个 加 括号 的 表达 式 最 外 层 肯定 有 一 对 括号 。 因 此 ， 我 们 可 以 这 么 做 : 
f(1^6|e6|1, true) = f(1 ^ (60|6|1), true) + 
f((1^6) | (80|1), true) + 
f((1^86|16) | 1, true) 
也 就 是 说 ， 我 们 可 以 迭代 整个 表达 式 ， 将 每 个 运算 符 当 作 第 一 个 要 加 括号 的 运算 符 。 
现在 ， 又 该 如 何 计算 这 些 内 层 的 表达 式 呢 ， 比 如 f((1*6) | (8|1)，true)? 很 简单 ， 要 让 
这 个 表达 式 的 值 为 true， 左 半 部 分 或 右 半 部 分 必 有 其 一 为 true。 因 此 ， 这 个 表达 式 分 解 如 下 : 
f((1^6) | (86|1), true) = f(1^8, true) * f(@|1, true) + 


f(1^8，false) * f(0|1, true) + 
f(1^0, true) * f(60|1, false) 
对 每 个 布尔 运算 符 ， 都 可 以 进行 类 似 的 分 解 : 
f(exp1 | exp2, true) = f(expl, true) * f(exp2, true) + 
f(exp1l, true) * f(exp2, false) + 
f(exp1，false) * f(exp2, true) 
f(exp1，true) * f(exp2, true) 
f(exp1l, true) * f(exp2, false) + 
f(exp1l, false) * f(exp2, true) 


对 于 false 结 果 ， 我 们 也 可 以 执行 非常 类 似 的 操作 : 


f(exp1l, false) * f(exp2, false) 
f(exp1l, false) * f(exp2, false) + 
f(exp1l, true) * f(exp2, false) + 
f(exp1l, false) * f(exp2, true) 
f(exp1l, true) * f(exp2, true) + 
f(exp1, false) * f(exp2, false) 





















































f(exp1 & exp2, true) 
f(exp1 ^ exp2, true) 


f(exp1 | exp2, false) 
f(exp1 & exp2, false) 


f(exp1 ^ exp2, false) 
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至 此 ， 要 解决 这 个 问题 ， 只 需 反复 套用 这 些 递 归 关系 即 可 。( 注意 : 为 了 避免 代码 行 不 必要 
的 回 绕 ， 以 及 确保 代码 的 可 读 性 ， 下 面 的 代码 使 用 了 非常 短 的 变量 名 。 ) 





1 public int f(String exp, boolean result, int s, int e) { 
2 if (s == e) { 

3 if (exp.charAt(s) == “1 && result) { 

4 return 1; 

5 } else if (exp.charAt(s) == “6 && Iresult) { 
6 return 1; 

7 } 

8 return 0; 

9 } 

16 int c = 0 

11 if (result) { 


12 for (int i = s+1;i<=e;i+=2){ 

13 char op = exp.charAt(i); 

14 if (op == ‘8&’) { 

15 c += f(exp, true, s, i - 1) * f(exp, true, i + 1, e); 
16 } else if (op == |”) { 

17 c += f(exp, true, s, i - 1) * f(exp, false, i + 1, e); 
18 c += f(exp, false, s, i - 1) * f(exp, true, i + 1, e); 
19 c += f(exp, true, s, i - 1) * f(exp, true, i + 1, e); 
26 } else if (op == “^) { 

21 c += f(exp, true, s, i - 1) * f(exp, false, i + 1, e); 
22 c += f(exp, false, s, i - 1) * f(exp, true, i + 1, e); 
23 } 

24 } 

25 } else { 

26 for (int i = s+1;i<=e;i+= 2) { 

27 char op = exp.charAt(i); 

28 if (op == ‘8&’) { 

29 c += f(exp, false, s, i - 1) * f(exp, true, i + 1, e); 
36 c += f(exp, true, s, i - 1) * f(exp, false, i + 1, e); 
31 c += f(exp, false, s, i - 1) * f(exp, false, i + 1,e); 
32 } else if (op == “|’) { 

33 c += f(exp, false, s, i - 1) * f(exp, false, i + 1,e); 
34 } else if (op == “人 ) { 

35 c += f(exp, true, s, i - 1) * f(exp, true, i + 1, e); 
36 c += f(exp, false, s, i - 1) * f(exp, false, i + 1,e); 
37 } 

38 } 

39 } 

46 return c; 

41 } 





虽然 这 么 做 可 行 ， 但 不 是 很 有 效 ， 对 于 同一 个 exp 的 值 ， 它 会 重 算 f(exp) 很 多 次 。 
要 解决 这 个 问题 ， 我 们 可 以 运用 动态 规划 ， 组 在 不 同 表 达 式 的 结果 。 注 意 ， 我 们 需要 根据 
expression 和 result 进 行 缓存 。 


1 public int f(String exp, boolean result, int s, int e， 


2 HashMap<String, Integer> q) { 
3 String key = “” + result + s+ e; 
4 if (q.containsKey(key)) { 
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5 return q.get(key); 

6 } 

7 

8 if (s == e) { 

9 if (exp.charAt(s) == ‘1’ && result == true) { 
16 return 1; 

1 } else if (exp.charAt(s) == “6 && result == false) { 
12 return 1; 

13 } 

14 return 0@; 

15 } 


16 int c = 6; 
17 if (result) { 


18 for (int i = s+1;i<=e;i+=2)1{ 

19 char op = exp.charAt(i); 

20 if (op == ‘8&”) { 

2 c += f(exp,true,s,i-1,q) * f(exp,true,i+1,e,q); 
22 } else if (op == |) { 

23 c += f(exp,true,s,i-1,q) * f(exp,false,i+1,e,q); 
24 c += f(exp,false,s,i-1,q) * f(exp,true,i+1,e,q); 
25 c += f(exp,true,s,i-1,q) * f(exp,true,i+1,e,q); 
26 } else if (op == ‘~^’”) { 

27 c += f(exp,true,s,i-1,q) * f(exp,false,i+1,e,q); 
28 c += f(exp,false,s,i-1,q) * f(exp,true,i+1,e,q); 
29 } 

36 } 

31 } else { 

32 for (int i = s+1;i<=e;i+= 2) { 

33 char op = exp.charAt(i); 

34 if (op == ‘8&’”) { 

35 c += f(exp,false,s,i-1,q) * f(exp,true,i+1,e,q); 
36 c += f(exp,true,s,i-1,q) * f(exp,false,i+1,e,q); 
37 c += f(exp,false,s,i-1,q) * f(exp,false,i+1,e,q); 
38 } else if (op == |) { 

39 c += f(exp,false,s,i-1,q) * f(exp,false,i+1,e,q); 
40 } else if (op == ‘^’”) { 

41 c += f(exp,true,s,i-1,q) * f(exp,true,i+1,e,q); 
42 c += f(exp,false,s,i-1,q) * f(exp,false,i+1,e,q); 
43 } 

44 } 

45 } 

46 q.put(key, c); 

47 return c; 

48 } 


Ti 


运用 动态 规划 后 , 虽然 该 算法 已 得 到 很 好 的 优化 , 但 还 不 够 最 优 。 要 是 知道 一 个 表达 式 有 多 
少 种 括号 的 放 法 ， 我 们 完全 可 以 借 由 total(exp) - f(exp = true) 来 算出 f(exp = false)。 

对 于 一 个 表达 式 有 几 种 括号 的 放 法 ， 的确 有 个 公式 解 ， 只 是 你 可 能 不 知道 黑 了 。 这 个 解 可 由 
卡 塔 兰 数 导 出 ， 其 中 n 为 运算 符 的 数目 : 














_ (2n)! 
” (n+DIn! 


经 过 这 次 调整 ， 实 现代 码 类 似 如 下 : 
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1 public int f(String exp, boolean result, int s, int e， 

2 HashMap<String, Integer> q) { 

3 String key = “” +Ss+ ej 

4 int c = 9) 

5 if (!q.containsKey(key)) { 

6 if (s == e) { 

7 if (exp.charAt(s) == 1) c= 1; 

8 else c = 0@; 

9 } 

16 

11 for (int i = s+1;i<=e;i+=2){ 

12 char op = exp.charAt(i); 

13 if (op == ‘8&’”) { 

14 c += f(exp,true,s,i-1,q) * f(exp,true,i+1,e,q); 
15 } else if (op == “|’) { 

16 int left_ops = (i-1-s)/2; // 括号 在 左边 

17 int right ops = (e -i - 1) / 2; // 括号 在 右边 
18 int total ways = total(left ops) * total(right_ops); 
19 int total _ false = f(exp,false,s,i-1,q) * 

26 f(exp,false,i+1,e,q); 

21 Cc += total ways - total false; 

22 } else if (op == ‘^’) { 

23 c += f(exp,true,s,i-1,q) * f(exp,false,i+1,e,q); 
24 c += f(exp,false,s,i-1,q) * f(exp,true,i+1,e,q); 
25 } 

26 } 

27 q.put(key, c); 

28 } else { 

29 c = q.get(key); 

36 

31 if (result) { 

32 return c; 

33 } else { 

34 int num ops = (e - s) / 2; 

35 return total(num ops) - c; 

36 

37 } 


9.10 ”扩展 性 与 存储 限制 


10.1 假设 你 正在 搭建 某 种 服务 ， 有 多 达 1000 个 客户 端 软件 会 调用 该 服务 ， 取 得 每 天 盘 后 
股票 价格 信息 〈 开盘 价 、 收 盘 价 、 最 高 价 与 最 低 价 )。 假 设 你 手 里 已 有 这 些 数据 ， 存 储 格式 可 自 
行 定义 。 你 会 如 何 设计 这 套 面向 客户 端的 服务 , 向 客户 端 软件 提供 信息 ?你 将 负责 该 服务 的 研发 、 
部 署 、 持 续 监 控 和 维护 。 描 述 你 想到 的 各 种 实现 方案 ， 以 及 为 何 推荐 采用 你 的 方案 。 该 服务 的 实 
现 技术 可 任 选 ， 此 外 ， 可 以 选用 任何 机 制 向 客户 端 分 发 信息 。( 第 72 页 ) 


解法 
从 此 题 描 述 来 看 , 我 们 要 关注 的 是 如 何 真正 地 将 信息 分 发 给 客户 端 。 在 此 假定 有 一 些 脚 本 可 
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以 神奇 地 把 信息 收集 起 来 。 


全 已 
用 


度 
实现 各 种 查询 。 此 外 ， 若 这 些 文件 有 新 增 数据 ， 可 能 会 打 乱 客户 端的 解析 机 制 。 


首先 ， 让 我 们 想 一 想 合乎 要 求 的 方案 应 该 具备 哪 几 方 面 。 

口 客户 端 软件 易 用 性 : 我 们 希望 这 套 服务 对 客户 端 实现 起 来 又 容易 又 好 用 。 

口 让 我 们 自己 实现 起 来 也 轻松 : 这 套 服务 应 该 是 越 容易 实现 越 好 ， 不 该 自 讨 苦 吃 ， 把 不 必 

要 的 工作 强加 到 自己 头 上 。 之 所 以 要 考虑 这 点 ， 不 仅 是 因为 研发 成 本 ， 还 有 维护 成 本 。 

口 灵活 应 对 未 来 需求 : 此 题 的 问 法 是 “在 现实 世界 中 你 会 怎么 做 "， 因 此 ， 我 们 应 该 从 解决 
实际 问题 的 角度 来 思考 。 理 想 情况 下 ， 我 们 不 想 受 到 实现 的 过 多 限制 ， 以 致 无 法 灵活 应 
对 条 件 或 需求 变更 。 

口 扩展 性 和 效率 : 关注 实现 方案 的 效率 ， 才 不 会 让 服务 负担 过 重 。 

有 了 这 些 注意 事项 ， 我 们 就 可 以 考虑 各 种 方案 了 。 



























































方案 1 
一 种 选择 是 ， 将 数据 直接 保存 在 纯 文本 文件 中 ， 证 客户 端 通过 菜 种 FTP 服 务 器 下 载 。 从 某 种 
来 说 , 这 么 做 容易 维护 ， 因 为 可 以 自如 地 查看 和 备份 这 些 文件 , 但 需要 更 复杂 的 文件 解析 才 
































方案 2 

我 们 可 以 使 用 标准 的 SQL 数 据 库 ， 让 客户 端 直接 接 入 。 这 么 做 有 如 下 优点 。 

口 如 需 支 持 新 功能 ， 这 种 做 法 提供 了 一 种 让 客户 端 查 询 和 处 理 数 据 的 简单 方式 。 例 如 ， 我 

们 可 以 轻松 、 高 效 地 执行 这 类 查询 : 返回 开盘 价 高 于 N 且 收盘 价 低 于 M 的 所 有 股票 。 

口 利用 标准 的 数据 库 功 能 就 能 提供 数据 回 滚 、 数 据 备份 和 各 种 安全 保障 。 我 们 不 必 做 无 谓 

的 重复 性 劳动 ， 因 此 实现 起 来 非常 轻松 。 

口 客户 端 可 以 很 容易 地 整合 现 有 应 用 。 在 各 种 软件 开发 环境 中 ，SQL 整 合 是 标准 功能 。 

那么 ， 使 用 SQL 数据 库 有 哪些 缺点 呢 ? 

口 相 比 我 们 真正 需要 的 , 它 所 造成 的 负担 过 重 。 为 了 提供 一 些 信息 , 我 们 并 不 一 定 需要 SQL 

后 端的 所 有 复杂 功能 。 

口 对 用 户 来 说 ， 数 据 库 基 本 不 可 读 ， 因 此 需要 多 一 层 实 现 ， 以 查看 和 维护 数据 。 而 这 会 增 

加 实现 成 本 。 

D 安全 性 : 尽管 SQL 数据 库 提 供 了 非常 明确 的 安全 等 级 , 我 们 还 是 要 谨慎 行事 , 不 让 客户 端 
存 取 它 们 不 该 访问 的 数据 。 此 外 ， 即 使 客户 端 不 会 有 “恶意 ”的 动作 ， 它 们 也 可 能 执行 
昂贵 和 低 效 的 查询 ， 而 我 们 的 服务 需 将 会 承担 这 些 开销 。 

列 出 这 些 缺 点 并 不 表示 我 们 不 该 使 用 SQL。 相反 , 列 出 它们 是 为 了 让 我 们 对 这 些 缺 点 心 知 肚 明 。 





















































方案 3 
就 分 发 信息 而 言 ，XML 也 是 一 种 不 错 的 选择 。 采 用 XML 时 ， 数 据 有 固定 的 格式 和 大 小 : 


company name (公司 名 )、open (开盘 价 )、high (最 高 价 )、low (最 低 价 )、closingPrice ( 收 
盘 价 )， 下 面 是 一 个 XML 格式 的 数据 样 例 : 
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1 《rooty> 

2 <date value=“2668-16-122> 

3 <company name="“foo”> 

4 <open>126.23</openy> 

5 <high>136.27</high> 

6 <low>122.83</low> 

7 <closingPrice>127.38</closingPrice> 
8 


</company> 
9 <company name="“bar”> 
16 <open>52.73</open> 
11 <high>66.27</high> 
12 <low>586.29</low> 
13 <closingPrice>54.91</closingPrice> 
14 </company> 
15 </date> 
16 <date value=“26068-16-11”> . . . </date> 
17 </root> 
这 种 做 法 有 如 下 优点 。 


口 容易 分 发 ， 也 容易 为 机 器 和 人 类 识别 。 这 也 是 XML 成 为 分 享 和 分 发 数据 的 标准 数据 模型 
的 原因 之 一 。 
口 大 多 数 语言 都 有 执行 XML 解析 的 库 ， 因 此 客户 端 实现 起 来 也 很 容易 。 
口 在 XML 文件 中 增加 新 结 点 就 可 以 添加 新 数据 。 这 不 会 打 乱 客户 端 解析 器 〈 只 要 以 正确 的 
方式 实现 解析 器 )。 
口 数据 以 XML 文件 格式 存储 ,因此 我 们 可 以 利用 现 有 工具 备份 数据 , 不 必 自 己 重 新 做 一 套 。 
这 么 做 可 能 有 以 下 缺点 。 
口 这 种 做 法 会 向 客户 端 发 送 所 有 信息 ， 即 使 他 们 只 需要 其 中 一 部 分 。 这 么 做 效率 很 低 。 
口 进行 数据 查询 时 ， 必 须 解析 整个 文件 。 

无 论 采 用 哪 种 数据 存储 方案 ， 我 们 都 可 以 提供 Web 服 务 〈 比如 SOAP ) 供 客户 端 存 取 数 据 。 
这 会 在 工作 中 多 加 一 层 ， 但 它 能 够 提供 额外 的 安全 性 ， 甚 至 还 可 能 使 客户 更 易 整合 系统 。 

话说 回来 , 这 有 利 也 有 次， 客户 端 将 只 能 按 我 们 预 设 或 希望 其 采用 的 方式 获取 数据 。 相 比 之 
下 ， 在 纯 SQL 实 现 中 ， 即 使 我 们 没有 预料 到 客户 端 需要 查询 最 高 股价 ， 它 们 还 是 可 以 进行 查询 。 

那么 ,该 采用 哪 种 方案 ? 这 里 并 没有 明确 的 答案 。 纯 文本 文件 方案 或 许 是 一 个 糟糕 的 选择 ， 
不 过 ， 对 于 SQL 或 XMI 方 案 ， 不 管用 不 用 Web 服 务 ， 你 都 可 以 摆 出 令 人 信服 的 理由 。 

这 类 问题 的 目的 不 是 看 你 能 否 得 出 “正确 ”答案 (并 没有 唯一 正确 的 答案 )， 而 是 看 你 如 何 
设计 一 个 系统 ， 怎 么 权衡 利 次 并 做 出 选择 。 

10.2 ”你 会 如 何 设计 诸如 Facebook 或 LinkedIn 的 超大 型 社交 网 站 ? 请 设计 一 种 算法 ， 展 
示 两 个 人 之 间 的 “连接 关系 ”或 “社交 路 径 ”( 比如 ,我 一 鲍 过 一 苏 瑚 一 杰 森 一 你 )。( 第 
73 页 ) 

解法 

这 个 问题 有 个 不 错 的 解法 ， 就 是 先 移 除 一 些 限制 条 件 ， 解 决 该 问题 的 简化 版 本 。 
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步骤 1: 简化 问题 一 一 先 忘 记 有 几 百 万 用 户 

首先 ， 让 我 们 忘掉 要 应 对 几 百 万 的 用 户 ， 针 对 简单 情况 设计 算法 。 

我 们 可 以 构造 一 个 图 , 每 个 人 看 作 一 个 结 点 ,两 个 结 点 之 间 耕 有 连 线 ， 则 表示 这 两 个 用 户 为 
朋友 。 


1 class Person { 





之 Person[] friends; 
3 // 其 他 信息 
4 } 


要 找到 两 个 人 之 间 的 连接 ， 可 以 从 其 中 一 个 人 开始 ， 直 接 进行 广度 优先 搜索 。 
为 什么 深度 优先 搜索 效果 不 彰 呢 ?因为 它 非常 低 效 。 两 个 用 户 可 能 只 有 一 度 之 隔 , 却 可 能 要 
在 他 们 的 “ 子 树 ” 中 搜索 几 百 万 个 结 点 后 ， 才 能 找到 这 条 非常 简单 而 直接 的 连接 。 


步骤 2: 处 理 数 百 万 的 用 户 

处 理 LinkedIn 或 Facebook 这 种 规模 的 服务 时 ， 不 可 能 将 所 有 数据 存放 在 一 台 机 器 上 。 这 就 意 
味 着 前 面 定义 的 简单 数据 结构 Person 并 不 管用 ， 朋 友 的 资料 和 我 们 的 资料 不 一 定 在 同一 台 机 器 
上 。 我 们 要 换 种 做 法 ， 将 朋友 列表 改 为 他 们 了 的 列表 ， 并 按 如 下 方式 追踪 。 

(1) 针对 每 个 朋友 ID ， 找 出 所 在 机 器 的 位 置 : int machine_index = getMachineIDForUser 
(personID ) ;。 

(2) 转 到 编号 为 #machine_index 的 机 器 。 

(3) 在 那 台 机 器 上 ， 执 行 : Person friend = getPersonWithID(person_id);。 

下 面 的 代码 描绘 了 这 一 过 程 。 我 们 定义 了 一 个 Server 类 , 包含 一 份 所 有 机 器 的 列表 , 还 有 一 
个 Machine 类 ， 代 表 一 台 单独 的 机 器 。 这 两 个 类 都 用 了 散 列 表 ， 从 而 有 效 地 查找 数据 。 


1 public class Server { 

2 HashMap<Integer, Machine> machines = 

3 new HashMap<Integer, Machine>(); 

4 HashMap<Integer, Integer> personToMachineMap = 
5 

6 

7 

8 











new HashMap<Integer, Integer>(); 


public Machine getMachineWithId(int machineID) { 
return machines.get(machineID); 


9 } 

106 

11 public int getMachineIDForUser(int personID) { 

12 Integer machineID = personToMachineMap.get(personID); 
13 return machineID == null ? -1 : machineID; 

14 } 

15 

16 public Person getPersonWithID(int personID) { 

17 Integer machineID = personToMachineMap.get(personID); 
18 if (machineID == null) return null; 

19 

20 Machine machine = getMachineWithId(machineID); 

21 if (machine == null) return null; 

22 

23 return machine.getPersonWithID(personID); 
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转 ， 


24 } 

25 } 

26 

27 public class Person { 

28 private ArrayList<Integer> friendIDs; 

29 private int personID; 

36 

31 public Person(int id) { this.personID = id; } 
32 


33 public int getID() { return personID; } 
34 public void addFriend(int id) { friends.add(id); } 


35 } 

36 

37 public class Machine { 

38 public HashMap<Integer, Person> persons = 
39 new HashMap<Integer, Person>(); 

46 public int machineID; 

41 

42 public Person getPersonWithID(int personID) { 
43 return persons.get(personID); 

44 } 

45 } 


其 实 还 有 更 多 的 优化 和 后 续 问 题 有 待 讨论 ， 下 面 是 其 中 的 一 些 想法 。 


优化 : 减少 机 器 间 跳 转 的 次 数 
从 一 台 机 咒 跳 转 到 另 一 台 机 顺 的 开销 很 昂贵 。 不 要 为 了 找到 某 个 朋友 就 在 机 器 之 间 任 意 跳 
而 是 试 着 批 处 理 这 些 跳 转 劲 作 。 举 例 来 说 ， 如 果 有 五 个 朋友 都 在 同一 台 机 器 上 , 那 就 应 该 一 

















次 怕 


E 找 出 来 。 


优化 : 智能 划分 用 户 和 机 器 
人 们 跟 生 活 在 同一 国家 的 人 成 为 朋友 的 可 能 性 比较 大 。 因此, 不 要 随意 将 用 户 划 分 到 不 同 机 





需 上 ， 而 应 该 尽量 按 国家 、 城 市 、 州 等 进行 划分 。 这 样 一 来 ， 就 可 以 减少 跳 转 的 次 数 。 





问题 ; 广度 优先 搜索 通常 要 求 "标记 ”访问 过 的 结 点 。 在 这 种 情况 下 你 会 怎么 做 ? 
在 广度 优先 搜索 中 ,通常 我 们 会 设 定 结 点 类 的 visited 标 志 ， 以 标记 访问 过 的 结 点 。 但 针对 此 








， 我 们 并 不 想 这 么 做 。 同 一 时 间 可 能 会 执行 很 多 搜索 操作 ， 因 此 直接 编辑 数据 的 做 法 并 不 妥当 。 





反之 ,我们 可 以 利用 散 列 表 模 仿 结 点 的 标记 动作 ， 以 查询 结 点 id， 看 它 是 否 访问 过 。 


其 他 扩展 问题 

口 在 真实 世界 中 ， 服 务 器 会 出 故障 。 这 会 对 你 造成 什么 影响 ? 

口 你 会 如 何 利用 缓存 ? 

口 你 会 一 直 搜 索 ， 直 到 图 的 终点 (无 限 ) 吗 ? 该 如 何 判断 何 时 放弃 ? 

口 在 现实 生活 中 ， 有 些 人 比 其 他 人 拥有 更 多 朋友 的 朋友 ， 因 此 更 容易 在 你 和 其 他 人 之 间 构 
建 一 条 路 径 。 该 如 何 利 用 该 数据 选择 从 哪里 开始 遍历 ? 

这 些 只 是 你 或 者 面试 官 可 能 会 提出 的 部 分 扩展 问题 ， 其 实 还 有 其 他 许多 问题 可 以 深入 讨论 。 
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10.3 ”给 定 一 个 输入 文件 ， 包 含 40 亿 个 非 负 整数 ， 请 设计 一 种 算法 ， 产 生 一 个 不 在 该 文件 
中 的 整数 。 假 定 你 有 1GB 内 存 来 完成 这 项 任务 。 


进 阶 
如 果 只 有 10MB 内 存 可 用 ， 该 怎么 办 9 (第 73 页 ) 
解法 


总 共 可 能 有 2” 或 40 亿 个 不 同 的 整数 ， 其 中 非 负 整数 共 2 个。 我 们 可 以 使 用 1GB 内 存 ， 或 者 
80 亿 个 比特 。 

这 样 一 来 , 用 这 80 亿 个 比特 , 就 可 以 将 所 有 整数 映射 到 可 用 内 存 的 不 同比 特 位 , 处 理 逻 辑 如 下 。 

(1) 创建 包含 40 亿 个 比特 的 位 向 量 (BV，bitvector )。 回想 一 下 ,位 向 量 其 实 就 是 数组 ,利用 
整数 (或 另 一 种 数据 类 型 ) 数组 紧凑 地 储存 布尔 值 。 每 个 整数 可 存储 一 串 32 比 特 或 布尔 值 。 

(2) 将 BV 的 所 有 元 素 初 始 化 为 0。 

(3) 扫描 文件 中 的 所 有 数字 (num )， 并 调用 BV. set (num，1)。 

(4) 接着 ， 再 次 从 索引 0 开始 扫描 Bv。 

(5) 返回 第 一 个 值 为 0 的 索引 。 

下 面 的 代码 示范 了 上 面 的 算法 。 














1 long numberOfInts = ((long) Integer.MAX VALUE) + 1; 

2 byte[] bitfield = new byte [(int) (numberOfInts / 8)]; 

3 void findOpenNumber() throws FileNotFoundException { 

4 Scanner in = new Scanner(new FileReader(“file.txt”)); 
5 while (in.hasNextInt()) { 

6 int n = in.nextInt (); 

7 /* 使 用 OR 操作 符 设置 一 个 字 节 的 第 n 位 ， 

8 * 找 出 bitfield 中 相对 应 的 数字 ， 


9 * (例如 ，168 将 对 应 于 字 节 数组 中 索引 2 

16 * 的 第 2 位 ) */ 

11 bitfield [n/ 8] |= 1 << (n % 8); 

1 } 

13 

14 for (int i = 60; i < bitfield.length; i++) { 
15 for (int j = 6;j j < 8; j++) { 

16 /* 取 回 每 个 字 节 的 各 个 比特 。 当 发 现 

17 * 某 个 比特 为 8 时 ， 即 找到 相对 应 的 值 */ 
18 if ((bitfield[i] & (1 << j)) == 6) { 
19 System.out.println (i * 8 + j); 
20 return; 

21 } 

22 } 

23 

24 } 


进 阶 ， 只 能 使 用 10MB 内 存 该 怎么 办 ? 
对 数据 集 进行 两 次 扫描 , 就 可 以 找 出 不 在 文件 中 的 整数 。 我 们 可 以 将 全 部 整数 划分 成 同等 大 
小 的 区 块 ( 稍 后 会 讨论 如 何 决 定 大 小 )。 这 里 假设 要 将 整数 划分 为 大 小 为 1000 的 区 块 。 那 么 ,区 
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块 0 代 表 0 ~ 999 的 数字 ， 区 块 1 代 表 1000 ~ 1999 的 数字 ， 依 此 类 推 。 
因为 所 有 数值 各 不 相同 ,我 们 很 清楚 每 个 区 块 应 该 有 多 少数 字 , 所 以 , 扫描 文件 时 ， 数 一 数 
0 ~ 999 之 间 有 多 少 个 值 ，1000 ~ 1999 之 间 有 多 少 个 值 ， 依 此 类 推 。 如 果 在 某 个 区 块 内 只 有 999 个 
值 ， 即 可 断定 该 范围 内 少 了 某 个 数字 。 
在 第 二 次 扫描 时 ,我们 要 真正 找 出 该 范围 内 少 了 哪个 数字 。 我 们 可 以 采用 先前 位 向 量 的 做 法 ， 
并 忽略 该 范围 之 外 的 任意 数字 。 
眼下 ， 问 题 在 于 区 块 多 大 才 合 适 ? 下面 先 定义 若干 变量 。 
口 令 rangeSize 为 第 一 次 扫描 时 每 个 区 块 的 范围 大 小 。 
口 今 arraySsize 表 示 第 一 次 扫描 时 区 块 的 个 数 。 注 意 ，arraysSize = 22 / rangeSize， 因 为 
一 共有 2” 个 整数 。 
我 们 需要 为 rangeSize 选 择 一 个 值 ， 以 使 第 一 次 扫描 (数组 ) 与 第 二 次 扫描 (位 向 量 ) 所 需 
的 内 存 够 用 。 
@ 第 一 次 扫描 : 数组 
第 一 次 扫描 所 需 的 数组 可 以 填 人 10MB 或 大 约 22 字 节 的 内 存 中 。 数 组 中 每 个 元 素 均 为 整数 
( int )， 而 每 个 整数 有 4 字 节 ， 因 此 可 以 使 用 最 多 包含 约 22 个 元 素 的 数组 。 综 上 ， 我 们 可 以 导出 如 
下 式 子 : 
@ 第 二 次 扫描 : 位 向 量 
232 


arraySize = 一 一 -一 芝 和 2 
rangeSize 














rangeSize 之 气 
rangeSize 过 2 
我 们 需要 有 足够 的 空间 储存 rangesize 个 比特 。 我 们 可 以 将 2 个 字 节 放 进 内 存 ， 自 然 就 能 存 
放 2”* 个 比特 。 因 此 ， 可 以 推出 如 下 式 子 : 
2 < rangeSize < 2 
在 这 些 条 件 下 ,我 们 有 足够 的 空间 回旋 , 但 是 如 果 挑 选 出 越 靠近 中 间 的 值 ， 那么 ,在 任何 时 
候 所 需 的 内 存 就 越 少 。 
下 面 的 代码 提供 了 该 算法 的 一 种 实现 。 
int bitsize = 1648576;j // 2^26 比 特 (2^17 字 节 ) 
int blockNum = 4696; // 2^12 


byte[] bitfield = new byte[bitsize/8]; 
int[] blocks = new int[blockNum]; 














void findopenNumber() throws FileNotFoundException { 
int starting = -1; 
Scanner in = new Scanner (new FileReader (“file.txt”)); 
9 while (in.hasNextInt()) { 
16 int n = in.nextInt(); 
11 blocks[n / (bitfield.length * 8)]++; 


ovamA 和 Fw 
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13 

14 for (int i = 6; i < blocks.length; i++) { 

15 if (blocks[i] < bitfield.length * 8){ 

16 /* 若 value < 2^286， 那么 该 区 段 里 至 少 

17 * 少 了 一 个 数字 */ 

18 starting = i * bitfield.length * 8; 

19 break; 

26 } 

21 } 

22 

23 in = new Scanner(new FileReader(“file.txt”)); 
24 while (in.hasNextInt()) { 

25 int n = in.nextInt(); 

26 /* 若 该 数字 落 在 少数 字 的 区 块 里 ， 

27 * 则 记 下 该 数字 */ 

28 if (n >= starting && n < starting + bitfield.length * 8) { 
29 bitfield [(n-starting) / 8] |= 1 << ((n - starting) % 8); 
36 } 

31 } 

32 

33 for (int i = 6 ; i < bitfield.length; i++) { 
34 for (int j = 6; j < 8; j++) { 

35 /* 取 回 每 个 字 节 的 各 个 比特 ， 当 发 现 有 比特 为 9 时 ， 
36 * 找到 相对 应 的 值 */ 

37 if ((bitfield[i] & (1 << j)) == 6) { 

38 System.out.println(i * 8 + j + starting); 
39 return; 

46 } 

41 } 

42 

43 } 


紧 接着 ,面试 官 可 能 还 会 问 你 ,可 用 内 存 更 少 的 话 ， 又 该 怎么 办 ? 在 这 种 情况 下 ,我 们 会 采 
用 第 一 步骤 的 做 法 重复 扫描 。 首 先 检 查 每 100 万 个 元 素 序 列 中 会 找到 多 少 个 整数 。 接 着 ， 在 第 二 
次 扫描 时 ,检查 每 1000 个 元 素 的 序列 中 可 找到 多 少 个 整数 。 最 后 ,在 第 三 次 扫描 时 , 使 用 位 向 量 
找 出 不 在 文件 中 的 那个 数字 。 


10.4 给 定 一 个 数组 ， 包 含 1 到 N 的 整数 ，N 最 大 为 32 000， 数 组 可 能 含有 重复 的 值 ， 且 N 
的 取 值 不 定 。 若 只 有 4KB 内 存 可 用 ， 该 如 何 打印 数组 中 所 有 重复 的 元 素 。( 第 73 页 ) 


解法 

我 们 有 4KB 内 存 可 用 ， 也 就 是 最 多 可 寻 址 8* 4* 2 个 比特 。 注 意 ，32* 210" 要 比 32 000 大 。 我 
们 可 以 创建 含有 32 000 个 比特 的 位 向 量 ， 其 中 每 个 比特 代表 一 个 整数 。 

利用 这 个 位 向 量 ， 就 可 以 迭代 访问 整个 数组 ， 发 现 数组 元 素 v 时 ， 就 将 位 v 设 定 为 1。 磁 到 重 
复元 素 时 ， 就 打印 出 来 。 


1 public static void checkDuplicates(int[] array) { 
2 BitSet bs = new BitSset(32666) ; 

3 for (int i = 86; i < array.length; i++) { 

4 

2 














int num = array[i]; 
int num@ = num - 1; // bitset 从 6 开始 ， 数 字 从 1 开始 
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6 if (bs.get(num6)) { 

区 System.out.println(Cnum); 
8 } else { 

9 bs.set(nume); 

16 } 

11 } 

Fp 

13 


14 class BitSet { 
15 int[] bitset; 


16 

17 public BitSet(int size) { 

18 bitset = new int[size >> 5]; // 除 以 32 

19 } 

20 

21 boolean get(int pos) { 

2 int wordNumber = (pos >> 5); // 除 以 32 

23 int bitNumber = (pos & 6x1F); // 除 以 32 取 余数 
24 return (bitset[wordNumber] & (1 << bitNumber)) != @; 
25 } 

26 

27 void set(int pos) { 

28 int wordNumber = (pos >> 5); // 除 以 32 

29 int bitNumber = (pos & 6x1F); // 除 以 32 取 余数 
36 bitset[wordNumber] |= 1 << bitNumber; 

31 } 

32 } 





注意 , 虽然 此 题 不 太 难 , 但 重要 的 是 实现 代码 要 写 得 干净 利落 。 这 也 是 为 什么 要 定义 位 向 量 
类 来 保存 大 型 的 位 向 量 。 要 是 面试 官 允 许 〈 也 可 能 不 会 )， 那 就 可 以 使 用 Java 内 置 的 Bitset 类 。 


10.5 “如果 要 设计 一 个 网 络 爬 虫 程序 ， 该 怎么 避免 陷入 无 限 循环 ?* (第 73 页 ) 


解法 

针对 此 题 , 第 一 个 要 问 自己 的 是 : 什么 情况 下 才 会 出 现 无 限 循环 ”最 直接 的 答案 是 ， 如 果 将 
整个 网 络 想 象 成 一 个 链接 的 图 ， 图 中 有 环 就 会 出 现 无 限 循环 。 

为 了 避免 无 限 循环 , 我 们 只 需 检测 有 没有 环 。 一 种 做 法 是 创建 一 个 散 列 表 , 访问 过 页 面 v 后 ， 
将 hash[v] 设 为 直 (true )。 

这 种 解法 意味 着 使 用 广度 优先 搜索 的 方式 抓 取 网 站 。 每 访问 一 个 页 面 , 我 们 就 会 收集 它 的 所 
有 链接 ， 并 将 它们 插入 队列 末尾 。 若 发 现 茶 个 页 面 已 访问 ， 就 将 其 忽略 。 

这 个 方法 不 错 ， 不 过 访问 页 面 v 意 味 着 什么 ?页 面 v 是 基于 它 的 内 容 还 是 URL 来 定义 的 ? 

如 果 页 面 是 根据 其 URL 定 义 的 ， 我 们 必须 认识 到 URL 参 数 可 能 代表 完全 不 同 的 页 面 。 例 如 ， 
页 面 www.careercup.com/page?id=microsoft-interview-questions 与 页 面 www.careercup. 
com/page?id=google-interview-questions 是 完全 不 一 样 的 。 不 过 ， 只 要 URL 参 数 不 是 Web 应 
用 识别 和 处 理 的 ， 就 可 以 将 它 附 加 到 任意 URL 之 后 ， 而 不 会 真 的 改变 页 面 ， 比 如 ， 页 面 


www.careercup.com?foobar=hello 与 www.careercup.com 是 一 样 的 。 
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“好 吧 ,” 你 或 许 会 说 ,“ 那 我 们 就 以 内 容 定义 页 面 。 乍 一 听 ， 似 乎 还 不 错 ， 但 这 并 不 切实 可 
行 。 假 设 careercup.com 首 页 的 部 分 内 容 是 随机 后 成 的 。 每 次 访问 首页 时 ， 它 都 是 不 同 的 页 面 吗 ? 
不 见得 。 

现实 情况 是 目前 还 没有 完美 的 方式 来 定义 “不 同 的 ”页 面 ， 这 就 是 此 题 坏 手 的 地 方 。 

一 种 解决 方法 是 评估 相似 程度 。 根 据 内 容 和 URL , 若 某 个 页 面 与 其 他 页 面具 有 一 定 的 相似 度 ， 
则 降低 抓 取 其 子 页 面 的 优先 级 。 对 于 每 个 页 面 , 我 们 都 会 根据 内 容 片 段 和 页 面 的 URL, 算出 某 种 
寺 征 码 。 

下 面 我 们 来 看 看 这 是 如 何 实现 的 。 

我 们 有 一 个 数据 库 , 储存 了 待 抓 取 的 一 系列 条 目 。 每 一 次 循环 ,我 们 都 会 选择 最 高 优先 级 的 
页 面 进 行 抓 取 ， 接 着 执行 以 下 步骤 。 

(1) 打开 该 页 面 ， 根 据 页 面 的 特定 片段 及 其 URL， 创建 该 页 面 的 特征 码 。 

(2) 查询 数据 库 ， 看 看 最 近 是 否 已 抓 取 拥 有 该 特征 码 的 页 面 。 

(3) 若 有 此 特征 码 的 页 面 最 近 已 被 抓 取 过 ， 则 将 该 页 面 插 回 数据 库 ， 并 调 低 优先 级 。 

(4) 若 未 抓 取 ， 则 抓 取 该 页 面 ， 并 将 它 的 链接 插入 数据 库 。 

根据 上 面 的 实现 , 我 们 怎么 也 “ 完 不 成 ”整个 Web 的 抓 取 , 但 可 以 避免 陷入 页 面 循环 的 情况 。 
阁 想 最 终 “ 完 成 ”整个 Web 的 抓 取 ( 显然， 只 有 当 这 个 “Web” 是 诸如 企业 内 部 网 那 种 较 小 的 系 
统 时 才 可 行 )， 那么 ， 可 以 设 定 一 个 保证 页 面 一 定 会 被 抓 取 的 最 低 优 先 级 。 

这 只 是 一 个 简化 的 解法 , 实际 上 还 有 许多 其 他 同样 有 效 的 解法 。 这 类 问题 更 像 是 你 跟 面 试 官 
之 间 的 对 话 ， 可 能 引发 出 各 种 各 样 的 讨论 。 事 实 上 ， 针 对 此 题 的 讨论 很 有 可 能 引出 下 一 题 。 


10.6 给 定 100 亿 个 网 址 ， 如 何 检测 出 重复 的 文件 ? 这 里 所 谓 的 ”重复 “是 指 两 个 URL 完 全 
相同 。( 第 73 页 ) 


解法 

100 亿 个 网 址 (URL ) 要 占用 多 少 空间 呢 ? 如 果 每 个 网 址 平均 长 度 为 100 个 字符 ， 每 个 字 
符 占 4 字 节 , 则 这 份 100 亿 个 网 址 的 列表 将 占用 约 4 兆 兆 字 节 (4TB )。 在 内 存 中 可 能 放 不 下 那么 
多 数据 。 

不 过 ， 不 妨 假装 一 下 ， 这 些 数据 真 的 奇迹 般 地 放 进 了 内 存 ， 毕 竟 先 求解 简化 的 题目 是 很 
有 用 的 做 法 。 对 于 此 题 的 简化 版 ， 只 要 创建 一 个 散 列 表 ， 若 在 网 址 列表 中 找到 某 个 URL， 就 
映射 为 true。( 男 一 种 做 法 是 对 列表 进行 排序 , 找 出 重复 项 , 这 需要 额外 耗费 一 些 时 间 , 但 几 
无 优点 可 言 。) 

至 此 ， 我 们 得 到 此 题 简化 版 的 解法 ,那么 ， 假 设 我 们 手 上 有 4000GB 的 数据 ， 而 且 无 法 全 
部 放 进 内 存 ， 该 怎么 办 ” 倒 也 好 办 ， 我 们 可 以 将 部 分 数据 储存 至 磁盘 ， 或 者 将 数据 分 拆 到 多 
台 机 器 上 。 


解法 1: 储存 至 磁盘 
若 将 所 有 数据 储存 在 一 台 机 器 上 ,可 以 对 数据 进行 两 次 扫描 。 第 一 次 扫描 是 将 网 址 列表 拆 分 
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为 4000 组 , 每 组 1GB 。 简 单 的 做 法 是 将 每 个 网 址 u 存 放 在 名 为 <x> .txt 的 文件 中 , 其 中 x = hash(u) 
%4666。 也 就 是 说 , 我 们 会 根据 网 址 的 散 列 值 ( 除 以 分 组 数量 取 余数 ) 分 割 这 些 网 址 。 这 样 一 来 ， 
所 有 散 列 值 相同 的 网 址 都 会 位 于 同一 文件 。 

第 二 次 扫描 时 , 我 们 其 实 是 在 实现 前 面 简 化 版 问题 的 解法 : 将 每 个 文件 载 和 内存 , 创建 网 址 
的 散 列 表 ， 找 出 重复 的 。 


解法 2: 多 台 机 器 

另 一 种 解法 的 基本 流程 是 一 样 的 ， 只 不 过 要 使 用 多 台 机 器 。 在 这 种 解法 中 , 我们 会 将 网 址 发 
送 到 机 器 x 上 ， 而 不 是 储存 至 文件 <x> .txt。 

使 用 多 台 机 器 有 优点 也 有 缺点 。 

主要 优点 是 可 以 并 行 执行 这 些 操作 ， 同 时 处 理 4000 个 分 组 。 对 于 海量 数据 ,这 么 做 就 能 迅速 
有 效 地 解决 问题 。 

缺点 是 现在 必须 依靠 4000 台 不 同 的 机 器 ， 同 时 要 做 到 操作 无 误 。 这 可 能 不 太 现 实 (特别 是 对 
于 数据 量 更 大 、 机 器 更 多 的 情况 )， 我 们 需要 开始 考虑 如 何 处 理 机 器 故障 。 此 外 ， 涉 及 这 么 多 机 
器 ， 无 疑 大 幅 增加 了 系统 的 复杂 性 。 

话说 回来 ， 这 两 种 解法 都 不 错 ， 都 值得 与 面试 官 讨论 一 番 。 


10.7 想象 有 个 Web 服 务 器 ， 实 现 简化 版 搜索 引擎 。 这 套 系 统 有 100 台 机 器 来 响应 搜索 查询 ， 
可 能 会 对 另外 的 机 器 集群 调用 processSearch(string query) 以 得 到 真正 的 结果 。 响 应 查询 请 求 的 
机 器 是 随机 挑选 的 ， 因 此 两 个 同样 的 请 求 不 一 定 由 同一 人 台 机 器 响应 。 方 法 processsearch 的 开销 很 
大 ， 请 设计 一 种 缓存 机 制 ， 缓 存 最 近 几 次 查询 的 结果 。 当 数据 发 生变 化 时 ， 务 必 说 明 该 如 何 更 新 
缓存 。( 第 73 页 ) 

解法 

在 开始 设计 系统 之 前 ， 必 须 先 理解 此 题 的 真正 含义 。 如 我 们 所 预料 的 ， 这 类 题目 有 很 多 细节 
都 比较 模糊 。 为 了 提供 一 个 解法 , 我们 将 做 出 一 些 合 理 的 假设 , 不 过 ,你 应 该 与 面试 官 深入 讨论 
这 些 细节 。 









































假设 

下 面 是 针对 这 个 解法 做 出 的 几 个 假设 条 件 。 基 于 系统 设计 和 解 题 的 方法 , 你 可 能 还 会 做 出 其 
他 假设 条 件 。 记 住 ， 虽 然 某 些 方法 会 比 其 他 的 好 一 些 ， 但 并 没有 唯一 “正确 ”的 方法 。 
口 除了 必要 时 往外 调用 processsearch， 所 有 查询 处 理 都 在 最 初 被 调用 的 那 台 机 器 上 完成 。 
口 我 们 希望 缓存 的 搜索 查询 数量 庞大 ( 几 百 万 )。 
口 机 器 之 间 的 调用 速度 相对 较 快 。 
口 给 定 查 询 的 结果 是 一 个 有 序 的 网 址 列表 ， 每 个 网 址 关联 50 个 字符 的 标题 和 200 个 字符 的 
摘要 。 
口 最 常见 的 查询 非常 热门 ， 以 至 于 它们 总 是 会 存在 缓存 中 。 
重申 一 次 ， 这 些 不 是 唯一 的 有 效 假设 ， 仅 是 其 中 几 个 合理 的 假设 。 
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系统 需求 


设计 缓存 机 制 时 ， 显 然 我 们 需要 支持 两 个 主要 功能 : 


口 
口 








给 定 茶 个 键 ， 快 速 有 效 地 查找 出 来 ; 
旧 的 数据 会 过 期 ， 从 而 让 它 可 被 新 的 数据 取代 。 


此 外 ,， 当 某 次 查询 的 结果 改变 时 , 我 们 还 必须 处 理 缓存 的 更 新 或 清除 。 因 为 有 些 查 询 非常 常 


己 





有 可 能 长 驻 在 缓存 中 ， 我 们 不 能 干 等 着 该 数据 过 期 。 


步骤 1: 设计 单 系统 的 缓存 
此 题 有 个 好 解法 : 先 针对 单 台 机 器 设计 缓存 。 那 么 ， 又 该 创建 什么 样 的 数据 结构 ,使 我 们 得 
以 轻易 清除 旧 数据 ， 还 能 高 效 地 根据 键 查找 出 相对 应 的 值 ? 


口 





口 











使 用 链表 可 以 轻易 清除 旧 数 据 ， 只 需 将 “新 鲜 ” 项 移 到 链表 前 方 。 当 链表 超过 一 定 大 小 
时 ， 我 们 可 以 删除 链表 末尾 的 元 素 。 
散 列表 可 以 高 效 查 找 数据 ， 但 通常 无 法 轻易 地 清除 数据 。 











怎样 才能 做 到 两 全 其 美 呢 ? 将 这 两 种 数据 结构 融合 在 一 起 即 可 ， 下 面 是 具体 做 法 。 

跟 之 前 一 样 创建 一 个 链表 ,每 次 访问 结 点 后 ， 这 个 结 点 就 会 移 至 链表 首部 。 这 样 一 来 ,链表 
尾部 将 总 是 包含 最 陈旧 的 信息 。 

此 外 ,还 需要 一 个 散 列表 , 将 查询 映射 为 链表 中 相应 的 结 点 。 这 样 不 仅 可 以 有 效 返 回 缓存 的 





士 四 
结果 ， 





还 能 将 适当 的 结 点 移 至 链表 首部 ， 从 而 更 新 其 “新 鲜 度 ”。 


为 了 说 明 这 种 方法 ,下 面 给 出 了 缩 略 的 缓存 实现 代码 。 本 书 网 站 提供 了 这 些 代码 的 完整 版 本 。 
注意 ， 在 面试 中 ,一般 不 会 要 求 你 为 此 写 出 完整 的 代码 ， 也 不 会 要 求 你 设计 更 大 的 系统 。 


1 
2 
4 
5 
6 
7 
8 


public class Cache { 
public static int MAX_SIZE = 10; 
public Node head, tail; 
public HashMap<String, Node> map; 
public int size = 0@; 


public Cache() { 
map = new HashMap<String, Node>(); 


上 


/* 将 结 点 移 至 链表 前 方 */ 
public void moveToFront(Node node) { ... } 
public void moveToFront(String query) { ... } 


/* 从 链表 中 移 除 结 点 */ 
public void removeFromLinkedList(Node node) { ... } 


/* 从 缓存 中 获取 结果 ， 并 更 新 链表 */ 
public String[] getResults(String query) { 
if (!map.containsKey(query)) return null; 


Node node = map.get(query); 


moveToFront(node); // 更 新 新 鲜 度 
Peturn node.results; 
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25 } 


27  ”/* 将 结果 插入 链表 ， 并 散 列 */ 
28 public void insertResults(String query, String[] results) { 


29 if (map.containsKey(query)) { // 更 新 值 
36 Node node = map.get(query); 

31 node.results = results; 

32 moveToFront(node); // 更 新 新 鲜 度 
33 return; 

34 } 

35 

36 Node node = new Node(query, results); 
37 moveToFront(node); 

38 map.put(query, node); 

39 

46 if (size > MAX_SIZE) { 

41 map.remove(tail.query); 

42 removeFromLinkedList(tail); 

43 } 

44 } 

45 } 


步骤 2: 扩展 到 多 台 机 器 

现在 , 我 们 了 解 了 如 何 设 计 单 台 机 器 的 缓存 , 接 下 来 还 需 了 解 ， 当 查询 被 发 送 至 许多 不 同 的 
机 器 时 ， 如 何 设计 缓存 。 回 想 一 下 问题 描述 : 不 能 保证 某 个 查询 一 定 会 发 送 给 同一 台 机 需 。 

首先 ， 我 们 需要 决定 缓存 跨 机 器 共享 到 什么 程度 。 有 以 下 几 种 选择 可 供 参 考 。 

@ 选择 1: 每 台 机 器 都 有 自己 的 缓存 

简单 的 选择 是 每 台 机 器 都 有 自己 的 缓存 。 也 就 是 说 ， 如 果 “foo” 在 短 时 间 内 被 发 送 给 机 器 1 
两 次 ， 在 第 二 次 ， 结 果 会 从 缓存 中 返回 。 但是， 如 果 “foo” 先 发 送 给 机 器 1 然后 发 送 至 机 器 2， 
则 两 次 都 会 被 视 作 全 新 的 查询 。 

这 么 做 的 优点 是 相对 快速 ,因为 不 涉及 机 器 之 间 的 调用 。 可 惜 ， 由 于 许多 重复 查询 都 会 被 视 
作 全 新 查询 ， 作 为 优化 工具 的 缓存 并 不 是 那么 有 效 。 

@ 选择 2: 每 台 机 器 都 有 一 个 缓存 的 副本 

另 一 个 极端 是 ,我 们 可 以 给 每 台 机 器 一 个 缓存 的 完整 副本 。 当 新 的 条 目 添加 至 缓存 时 ,它们 
会 被 发 送 给 所 有 机 器 。 包 括 链接 和 散 列 表 在 内 的 整个 数据 结构 都 会 被 复制 。 

这 种 设计 意味 着 常见 的 查询 几乎 总 是 会 在 缓存 里 ， 因 为 所 有 机 器 的 缓存 都 是 相同 的 。 但 是 ， 
主要 的 缺点 是 更 新 缓存 意味 着 要 将 数据 发 送 给 X 台 机 器 ， 其 中 N 是 响应 集群 的 规模 。 此 外 ， 每 个 
条 目 占 用 的 空间 是 上 一 种 做 法 的 N 倍 ， 因 此 缓存 所 能 存放 的 数据 要 少 得 多 。 

@ 选择 3: 每 台 机 器 储存 一 部 分 缓存 

第 三 种 选择 是 将 缓存 分 割 开 ， 每 台 机 器 存放 缓存 的 不 同 部 分 。 然 后 ， 当 机 器 ;需要 查找 某 次 
查询 的 结果 时 ， 它 会 算出 哪 一 台 机 器 持 有 这 个 值 ， 接 着 请 求 这 台 机 器 (机 器 / ) 在 它 的 缓存 里 查 
找 该 查询 。 

但 是 ， 机 器 ;怎么 知道 哪 一 台 机 器 持 有 这 部 分 散 列 表 ? 
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一 种 选择 是 根据 算式 hash(query) % N 指 定 查 询 的 结果 。 然 后 ,机 器 ;只 需 利 用 这 个 算式 即 可 得 
出 储存 结果 的 机 器 7。 

因此 ， 当 新 的 查询 进入 机 器 时， 这 人 台 机 器 会 应 用 上 面 的 算式 从 而 调用 机 器 7。 随后 ， 机 妖 j 
会 从 它 的 缓存 中 返回 竺 查询 的 值 ， 或 者 调用 processSsearch(query) 得 到 结果 。 机 器 /会 更 新 其 组 
存 ， 并 将 结果 返回 给 机 器 i。 

或 者 ,你 也 可 以 这 样 设计 系统 : 机 器 /在 其 当前 缓存 中 找 不 到 查询 的 结果 ,， 则 直接 返回 nul1。 
这 就 要 求 机 器 ;调用 processsearch， 然 后 将 结果 转发 给 机 器 /储存 。 这 个 实现 实际 上 会 增加 机 顺 
与 机 器 间 的 调用 数量 ， 没 什么 优势 可 言 。 


步骤 3: 内 容 改 变 时 更 新 结果 

回想 一 下 ， 有 些 查询 可 能 非常 热门 ， 以 致 缓存 足够 大 的 话 ， 它 们 可 能 会 永久 存在 缓存 中 。 当 
某 些 内 容 改 变 时 ， 我 们 需要 通过 某 种 机 制 来 定期 或 “ 按 需 ”刷新 缓存 的 结果 。 

要 回答 这 个 问题 ， 我 们 需要 考虑 结果 何 时 才 会 改变 (最 好 跟 面 试 官 讨 论 一 下 )。 结 果 改 变 的 
主要 时 机 如 下 。 

(1) 网 址 对 应 的 内 容 变 了 (或 网 址 对 应 的 页 面 被 移 除 )。 

(2) 为 反映 页 面 排名 变化 ， 搜 索 结 果 的 排序 也 变 了 。 

(3) 特定 查询 出 现 了 新 页 面 。 

为 了 处 理 情况 (1) 和 情况 (2)， 可 以 另外 创建 一 个 散 列 表 ， 指 示 哪 个 缓存 查询 与 特定 网 址 关联 。 
这 些 缓存 可 以 完全 独立 于 其 他 缓存 进行 处 理 ， 并 放 在 不 同 的 机 器 上 。 不 过 , 这 种 解法 可 能 需要 大 
量 的 数据 。 

另外 ， 如 果 数 据 不 要 求 即 时 刷新 〈 一 般 来 说 不 需要 )， 我 们 可 以 定期 遍历 每 台 机 器 上 储存 的 
缓存 ， 将 与 更 新 过 的 网 址 相关 联 的 结果 清除 掉 。 

情况 (3) 很 难处 理 。 我 们 可 以 通过 解析 新 网 址 对 应 的 内 容 并 从 缓存 中 清除 这 些 单一 词 的 查询 ， 
来 更 新 单一 词 查询 。 不 过 ， 这 仅 能 处 理 单一 词 的 查询 。 
情况 G)( 或 我 们 要 处 理 的 其 他 类 似 情 况 ) 有 个 不 错 的 处 理 方式 , 就 是 实现 缓存 的 “自动 逾期 ”。 
也 就 是 说 ， 我 们 会 强加 一 个 超时 ， 任 何 一 个 查询 ， 不 管 它 有 多 热门 ， 都 无 法 在 缓存 中 存放 超过 x 
分 钟 。 这 将 确保 所 有 的 数据 都 会 定期 刷新 。 


步骤 4: 继续 改进 

根据 你 做 出 的 假设 和 想 要 优化 的 情况 ， 这 个 设计 还 可 以 有 不 少 可 改进 和 优化 之 处 。 

其 中 有 个 优化 是 更 好 地 支持 有 些 查 询 非常 热门 的 情况 。 例 如 , 假设 ( 举 个 极端 的 例子 ) 所 有 
查询 中 ， 有 1% 都 含有 某 个 字符 串 。 那 么 ， 机 器 不必 每 次 都 将 这 个 搜索 请 求 转 给 机 器 /， 应 该 只 向 
j 转 发 一 次 ， 然 后 机 器 ;就 可 以 直接 将 结果 储存 在 自己 的 缓存 中 。 

或 者 , 我 们 还 可 以 重新 架构 整个 系统 , 根据 查询 的 散 列 值 而 不 是 随机 将 查询 分 配给 某 台 机 器 
( 由 此 也 得 到 缓存 的 位 置 )。 不 过 ， 这 么 做 也 有 利 有 弊 。 

另 一 个 优化 是 针对 “自动 过 期 ”机 制 的 。 按 照 前 面 的 描述 ， 这 个 机 制 会 在 X 分 钟 后 清除 任意 
数据 。 然 而 ， 相 比 其 他 数据 〈 如 历史 股价 )， 我 们 和 硕 望 茶 些 数据 〈 如 时 事 新 闻 ) 的 更 新 更 频繁 ， 
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可 以 根据 主题 或 网 址 实现 不 同 的 自动 逾期 机 制 。 对 于 后 一 种 情况 ,根据 页 面 以 往 的 更 新 频 度 ， 每 
个 网 址 会 设置 不 同 的 超时 值 。 该 搜索 查询 的 超时 值 是 每 个 网 址 超时 值 的 最 小 值 。 

这 只 是 一 部 分 可 以 改进 的 地 方 。 记 住 , 这 类 题 型 并 没有 唯一 正确 的 解法 ,其 用 意 是 让 你 与 面 
试 官 讨论 设计 准则 ， 展 示 你 的 思考 方式 和 人 解 题 方法 。 


9.11 ”排序 与 查找 
































11.1 给 定 两 个 排序 后 的 数组 A 和 B ， 其 中 A 的 末端 有 足够 的 缓冲 空 容纳 B。 编 写 一 个 方法 ， 
将 B 合 并 入 A 并 排序 。( 第 77 页 ) 


解法 

已 知 数组 A 未 端 有 足够 的 缓冲 ， 不 需要 再 分 配额 外 空间 。 程 序 的 处 理 逻 辑 很 简单 ， 就 是 逐一 
比较 A 和 B 中 的 元 素 ， 并 按 顺 序 插入 数组 ， 直 至 耗 尽 A 和 B 中 的 所 有 元 素 。 

这 么 做 的 唯一 问题 是 ， 如 果 将 元 素 插 入 数组 A 的 前 端 ， 就 必须 将 原 有 的 元 素 往 后 移动 ， 以 腾 
出 空间 。 更 好 的 做 法 是 将 元 素 插 入 数组 A 的 末端 ， 那 里 都 是 空闲 的 可 用 空间 。 

下 面 的 代码 就 实现 了 上 述 做 法 ， 从 数组 A 和 B 的 末端 元 素 开始 ， 将 最 大 的 元 素 放 到 数组 A 
的 末端 。 














1 public static void merge(int[] a, int[] b, int lastA, int lastB) { 
2 int indexA = lastA - 1; /* 数组 a 最 后 元 素 的 索引 */ 

3 int indexB = lastB - 1; /* 数组 b 最 后 元 素 的 索引 */ 

4 int indexMerged = lastB + lastA - 1; /* 合并 后 数组 的 最 后 元 素 索 引 */ 
5 

6 /* 合并 a 和 b， 从 这 两 个 数组 的 最 后 元 素 开始 */ 

2 while (indexA >= 6 && indexB >= 86) { 

8 /* 数组 a 最 后 元 素 > 数组 b 最 后 元 素 */ 

9 if (a[indexA] > b[indexB]) { 

16 a[indexMerged] = a[indexA]; // 复制 元 素 

1 indexMerged--; // 更 新 索引 

12 indexA--; 

13 } else { 

14 a[indexMerged] = b[indexB]; // 复制 元 素 

15 indexMerged--; // 更 新 索引 

16 indexB--; 

17 } 

18 } 

19 


26 /* 将 数组 b 剩 余 元 素 复制 到 适当 的 位 置 */ 
21 while (indexB >= 6) { 


22 a[indexMerged] = b[indexB]; 
23 indexMerged--; 

24 indexB--; 

25 } 

26 } 





注意 ， 处 理 完 B 的 剩余 元 素 后 ， 你 不 需要 复制 A 的 剩余 元 素 ， 因 为 这 些 元 素 已 经 在 那里 了 。 9 
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11.2 ”编写 一 个 方法 ， 对 字符 串 数组 进行 排序 ， 将 所 有 变 位 词 排 在 相 邻 的 位 置 。( 第 77 页 ) 


解法 
此 题 有 个 要 求 ， 对 数组 中 的 字符 串 进行 分 组 ,将 变 位 词 排 在 一 起 。 注 意 ， 除 此 之 外 , 并 没有 





要 求 这 些 词 按 特 定 顺 序 排列 。 




















做 法 之 一 就 是 套用 一 种 标准 排序 算法 ， 比 如 归并 排序 或 快速 排序 ， 并 修改 比较 器 





( comparator )。 这 个 比较 器 用 来 指示 两 个 字符 串 互 为 变 位 词 就 是 相等 的 。 





检查 两 个 词 是 否 为 变 位 词 , 最 简单 的 方法 是 什么 呢 ? 我 们 可 以 数 一 数 每 个 字符 串 中 各 个 字符 


出 现 的 次 数 ， 两 者 相同 则 返回 true。 或 者 ， 直 接 对 字符 串 进 行 排序 ， 若 两 个 字符 串 互 为 变 位 词 ， 
排序 后 就 相同 。 








比较 器 的 实现 代码 如 下 。 


1 public class AnagramComparator implements Comparator<String> { 
2 public String sortChars(String s) { 

3 char[] content = s.toCharArray(); 

4 Arrays.sort(content); 

5 return new String(content); 

6 } 

7 

8 public int compare(String s1, String s2) { 

9 return sortChars(s1).compareTo(sortChars(s2)); 
16 } 

11 } 


下 面 ， 利 用 这 个 compareTc 方 法 而 不 是 一 般 的 比较 器 对 数组 进行 排序 。 

12 Arrays.sort(array, new AnagramComparator()); 

这 个 算法 的 时 间 复 杂 度 为 O(n log(n))。 

这 可 能 是 使 用 通用 排序 算法 所 能 取得 的 最 佳 情 况 了 , 但 实际 上 , 并 不 需要 对 整个 数组 进行 排 








厅 


局 


只 需 将 变 位 词 分 组 放 在 一 起 即 可 。 
我 们 可 以 使 用 散 列 表 做 到 这 一 点 ， 这 个 散 列 表 会 将 排序 后 的 单词 映射 到 它 的 一 个 变 位 词 列 





表 。 举 例 来 说 ， acre 会 映射 到 列表 {acre, race，care}。 一 旦 将 所 有 同 为 变 位 词 的 单词 分 组 在 
一 起 ， 就 可 以 将 它们 放 回 到 数组 中 。 


下 面 是 该 算法 的 实现 代码 。 


public void sort(String[] array) { 
Hashtable<String, LinkedList<String>> hash = 
new Hashtable<String, LinkedList<String>>(); 


for (String s : array) { 
String key = sortChars(s); 
if (!hash.containsKey(key)) { 


1 

2 

3 

4 

5 /* 将 同 为 变 位 词 的 单词 分 在 同一 组 */ 

6 

7 

8 

9 hash.put(key，new LinkedList<String>()); 








Q@ 由 变换 某 个 词 或 短语 的 字母 顺序 构成 的 新 的 词 或 短语 。 例 如 , “triangle” 是 “integral” 的 变 位 词 。 一 一 译 者 注 
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16 

11 LinkedList<String> anagrams = hash.get(key); 
12 anagrams .push(s); 

13 } 

14 


15 /* 将 散 列表 转换 为 数组 */ 
16 int index = 0@; 
17 for (String key : hash.keySet()) { 


18 LinkedList<String> list = hash.get(key); 
19 for (String t : list) { 

26 array[index] = 七 ; 

21 index++; 

22 } 

23 } 

24 } 

















你 或 许 看 出 来 了 ， 上 面 的 算法 是 从 桶 排序 法 修改 而 来 的 。 


11.3 ”给 定 一 个 排序 后 的 数组 ， 包 含 " 个 整数 ， 但 这 个 数组 已 被 旋转 过 很 多 次 ， 次 数 不 详 。 
请 编写 代码 找 出 数组 中 的 某 个 元 素 。 可 以 假定 数组 元 素 原先 是 按 从 小 到 大 的 顺序 排列 的 。( 第 
77 页 ) 


解法 

你 是 不 是 觉得 此 题 要 用 到 二 分 查找 法 ? 没 错 。 

在 经 典 二 分 查找 法 中 , 我们 会 将 x 与 中 间 元 素 进 行 比较 , 以 确定 x 属于 左 半 部 分 还 是 右 半 部 分 。 
此 题 的 复杂 之 处 在 于 数组 被 旋转 过 了 ， 可 能 有 一 个 拐点 。 以 下 面 两 个 数组 为 例 ; 

Array1: {16，15，26， 6， 5} 

Array2: {56， 5，26，36，461} 

这 两 个 数组 的 中 间 元 素 都 是 20， 但 5 在 其 中 一 个 数组 的 左边 ,在 另 一 个 的 右边 。 因 此 ， 只 将 x 
与 中 间 元 素 进行 比较 是 不 够 的 。 

不 过 ,如果 再 仔细 观察 一 下 ,就 会 发 现 数组 有 一 半 (左边 或 右边 ) 必定 是 按 正 常 顺序 (升序 ) 
排列 的 。 因 此 ， 我们 可 以 看 看 按 正常 顺序 排列 的 那 一 半数 组 ， 确 定 应 该 搜索 左 半边 还 是 右 半边 。 

例如 ， 如 果 要 在 Array1 中 查找 5， 我 们 可 以 比较 左 侧 元 素 ( 10 ) 和 中 间 元 素 ( 20 )。 由 于 10< 
20， 左 半边 一 定 是 按 正常 顺序 排列 的 。 男 外 ， 由 于 5 不 在 这 两 个 元 素 之 间 ， 因 此 接 下 来 应 该 搜索 
右 半 边 。 

在 Array2 中 ， 可 以 看 到 50 > 20， 因 此 右 半 边 必定 是 按 正常 顺序 排列 的 。 接 着 查看 中 间 元 素 
( 20 ) 和 右 侧 元 素 ( 40 ), 检查 5 是 否 落 在 这 两 个 元 素 之 间 。 显 然 5 并 不 落 在 两 者 之 间 ， 因 此 接 下 来 
要 搜索 右 半边 。 

如 果 左 侧 元 素 和 中 间 元 素 完全 相同 ， 比 如 数组 {人 2，2，2，3，4，2}， 这 种 情况 就 比较 复杂 了 。 
这 里 我 们 可 以 检查 最 右边 的 元 素 是 否 不 同 。 若 不 同 ， 可 以 只 搜索 右 半边 ， 否 则 ， 两 边 都 得 搜索 。 

























































































1 public int search(int a[], int left, int right, int x) { 
2 int mid = (left + right) / 2; 
3 if (x == a[mid]) { // 找到 元 素 
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4 Peturn mid; 

5 } 

6 if (right < left) { 
4 return -1; 

8 


} 


16 /* 左 半边 或 右 半边 必 有 一 边 是 按 正 常 顺 序 排 列 ， 
11 * 找 出 是 哪 一 半边 ， 然 后 利用 按 正常 顺序 排列 的 





12 * 半边 ， 确 定 该 搜索 哪 一 边 */ 

13 if (a[left] < a[mid]) { // 左 半 边 为 正常 排序 

14 if (x >= a[left] && x <= a[mid]) { 

15 return search(a,，left, mid - 1，Xx); // 搜索 左 半边 

16 } else { 

17 return search(a，mid + 1，right，x); // 搜索 右 半 边 
18 } 

19 } else if (a[mid] < a[left]) { // 右 半 边 为 正常 排序 

20 if (x >= a[mid] && x <= a[right]) { 

21 return search(a，mid + 1，right,，x); // 搜索 右 半 边 
2 } else { 

23 return search(a，left,，mid - 1，Xx); // 搜索 左 半边 

24 } 

25 } else if (a[left] == a[mid]) { // 左 半 边 都 是 重复 元 素 

26 if (a[mid] != a[right]) { // 若 右 边 元 素 不 同 ， 则 搜索 那 一 边 
27 return search(a，mid + 1，right,，x); // 搜索 右 半 边 
28 } else { // 否则 ， 两 边 都 得 搜索 

29 int result = search(a,，left, mid - 1，x); // 搜索 左 半边 
36 if (result == -1) { 

31 return search(a，mid + 1，right,，x); // 搜索 右 半 边 
32 } else { 

33 return result; 

34 } 

35 } 

36 } 

37 return -1; 

38 } 


若 所 有 元 素 都 不 同 ， 则 上 述 代码 执行 的 时 间 复 杂 度 为 O(log n)。 有 很 多 元 素 重复 的 话 ,算法 
时 间 复 杂 度 则 为 COD)。 因 为 若 有 很 多 重复 元 素 ， 数 组 〈 或 子 数组 ) 的 左 半 边 和 右 半边 往往 都 得 
查找 。 

注意 ,尽管 此 题 并 不 是 太 难 理解 ,但 要 完美 无 瑕 地 实现 却 很 难 。 实 现时 难免 会 犯错 , 不 必 太 
自足。 因为 很 容易 就 犯 差 一 错误 和 其 他 不 易 察觉 的 错误 , 所 以 , 务必 对 代码 进行 全 面 彻底 的 测试 。 


11.4 设想 你 有 一 个 20GB 的 文件 ， 每 一 行 一 个 字符 串 。 请 说 明 将 如 何 对 这 个 文件 进行 排 
序 。( 第 77 页 ) 

解法 

当面 试 官 给 出 20GB 大 小 的 限制 时 ， 实 际 上 在 暗示 些 什 么 。 就 此 题 而 言 ， 这 表明 他 们 不 希望 
你 将 数据 全 部 载 人 内 存 。 

该 怎么 办 呢 ?” 做 法 是 只 将 部 分 数据 载 入 内 存 。 
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我 们 将 把 整个 文件 划分 成 许多 块 ， 每 个 块 x MB， 其 中 x 是 可 用 的 内 存 大 小 。 每 个 块 各 自 进行 
排序 ， 然 后 存 回 文件 系统 。 

各 个 块 一 旦 完成 排序 ， 我 们 便 将 这 些 块 逐一 合并 在 一 起 ， 最 终 就 能 得 到 全 都 排 好 序 的 文件 。 

这 个 算法 被 称 为 外 部 排序 〈external sort )。 


11.5 有 个 排序 后 的 字符 串 数 组 ， 其 中 散布 着 一 些 空 字符 串 ， 编 写 一 个 方法 ， 找 出 给 定 字 
符 串 的 位 置 。( 第 77 页 ) 


解法 

如 果 没 有 那些 空 字符 串 ， 就 可 以 直接 使 用 二 分 查找 法 。 比 较 待 查找 字符 串 str 和 数组 的 中 间 
元 素 ， 然 后 继续 搜索 下 去 。 

针对 数组 中 散布 一 些 空 字 符 串 的 情形 , 我 们 可 以 对 二 分 查找 法 稍 作 修改 , 所 需 的 修改 就 是 与 
mid 进 行 比较 的 地 方 ， 如 果 mid 为 空 字符 串 ， 就 将 mid 换 到 离 它 最 近 的 非 空 字符 串 的 位 置 。 

下 面 以 递归 方式 解决 此 题 , 稍 加 修改 ,就 可 以 迭代 方式 实现 。 本 书 可 下 载 的 代码 里 提供 了 过 
代 实 现 。 




















1 public int searchR(String[] strings, String str, int first, 
2 int last) { 

3 if (first > last) return -1; 

4 /* 将 mid 移 到 中 间 */ 

5 int mid = (last + first) / 2; 

6 

7 /* 车 mid 为 空 字符 事 ， 找 出 离 它 最 近 的 非 空 字 符 事 */ 

8 if (strings[mid].isEmpty()) { 

9 int left = mid - 1; 

16 int right = mid + 1; 

11 while (true) { 

12 if (left < first && right > last) { 

13 return -1; 

14 } else if (right <= last && !strings[right].isEmpty()) { 
15 mid = right; 

16 break; 

17 } else if (left >= first && !strings[left].isEmpty()) { 
18 mid = left; 

19 break; 

26 } 

21 right++; 

22 left--; 

23 } 

24 } 

25 


26 /* 检查 字符 串 ， 如 有 必要 则 继续 递归 */ 
27 if (str.equals(strings[mid])) { // 找到 了 


28 return mid; 

29 } else if (strings[mid].compareTo(str) < 6) { // 搜索 右 半 边 
36 return searchR(strings, str, mid + 1, last); 

31 ”} else { // 搜索 左 半 边 

32 return searchR(strings, str, first, mid - 1); 
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O(n) 


下 ; 


每 一 
这 个 


于 55 
不 可 
素 大 


必须 


36 public int search(String[] strings, String str) { 
37 if (strings == null || str == null || str == “”) { 


38 return -1; 

39 } 

46 return searchR(strings，str，6，strings.length - 1); 
41 } 





如 果 要 查找 空 字符 串 ， 务 必 小 心 对 待 。 我 们 该 找 出 空 字符 串 的 位 置 ( 该 操作 时 间 复 杂 度 为 
) ? 还 是 应 该 把 这 种 情形 作为 错误 处 理 ? 

很 遗憾 , 这 里 并 没有 正确 的 答案 。 关 于 这 一 点 你 应 该 与 面试 官 进 行 讨论 ， 只 需 简 单 地 询问 一 
就 能 表明 你 是 个 细心 的 程序 员 。 

11.6 “给 定 MxNA 和 6 阵 ， 每 一 行 、 每 一 列 都 按 升序 排列 ， 请 编写 代码 找 出 某 元 素 。( 第 77 页 ) 
解法 

解法 1 

在 第 一 种 方法 里 ， 我 们 可 以 对 每 一 行进 行 二 分 查找 ， 以 找到 元 素 在 哪 。 该 矩阵 有 M 行 ， 搜 索 
行 用 时 OUlogCV)) , 因此 这 个 算法 的 时 间 复 杂 度 为 O(M 1og(N)), 在 你 开始 构思 更 好 的 算法 之 前 ， 
算法 值得 向 面试 官 一 提 。 

要 设计 一 个 算法 ,我 们 先 从 一 个 简单 的 例子 开始 。 




















15 [| 20 [ 40 | 85 | 
20 | 35 | 80 | 95 | 
30 | 55 | 95 | 105 | 
40 | 8e | lee | 120 


假设 要 查找 元 素 55， 该 如 何 找 出 它 在 哪儿 呢 ? 

只 要 看 看 一 行 或 一 列 的 起 始 元 素 , 我 们 就 能 开始 推断 待 查 元 素 的 位 置 。 若 一 列 的 起 始 元 素 大 
， 就 表示 55 不 可 能 在 那 一 列 ， 因 为 起 始 元 素 是 那 一 列 的 最 小 元 素 。 此 外 ， 我 们 也 可 推断 55 
能 在 那 一 列 的 右边 ， 因 为 每 一 列 的 第 一 个 元 素 从 左 到 右 依次 增 大 。 因 此 , 若 那 一 列 的 起 始 元 
于 待 查找 的 元 素 x， 就 能 确定 我 们 必须 往 那 一 列 的 左边 查找 。 

对 于 和 矩阵 的 行 来 说 ， 可 以 套用 同样 的 逻辑 。 阁 某 一 行 的 起 始 元 素 大 于 x， 就 应 该 往 上 查找 。 
同样 地 ， 我 们 也 可 以 从 列 或 行 的 末端 得 出 类 似 的 结论 ， 若 某 一 列 或 行 的 未 尾 元 素 小 于 x， 就 
往 下 ( 行 ) 或 往 右 ( 列 ) 查找 ， 这 是 因为 末尾 元 素 必定 是 最 大 的 元 素 。 

下 面 我 们 可 以 将 这 些 观 察 到 的 要 点 合并 成 一 个 解法 ， 观 察 到 的 要 点 包括 : 

口 若 列 的 开头 大 于 x， 那 么 x 位 于 该 列 的 左边 ; 

口 若 列 的 末端 小 于 x， 那 么 x 位 于 该 列 的 右边 ; 

口 奉行 的 开头 大 于 x， 那 么 x 位 于 该 行 的 上 方 ; 
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口 若 行 的 末端 小 于 x， 那 么 x 位 于 该 行 的 下 方 。 

我 们 可 以 从 任意 位 置 开始 搜索 ， 不 过 ， 让 我 们 从 列 的 起 始 元 素 开始 。 

我 们 需要 从 最 大 的 那 一 列 开始 ， 然 后 向 左 移动 ， 这 意味 着 第 一 个 要 比较 的 元 素 是 
array[8][c-1]， 其 中 c 为 列 的 数目 。 将 各 个 列 的 开头 与 x ( 这 里 为 55 ) 进行 比较 ,就 会 发 现 x 必 定 
位 于 列 0、 列 1 或 列 2， 比 较 至 array[8][2] 停 下 来 。 

这 个 元 素 不 一 定 会 在 完整 矩阵 的 某 一 列 的 末端 , 但 会 是 某 个 子 矩 阵 的 某 一 列 的 末端 。 同 样 的 
条 件 一 样 适用 ，array[86][2] 的 值 是 40， 比 55 小 ， 由 此 可 知 必须 往 下 移动 。 

现在 ,我 们 以 下 面 这 个 子 和 矩阵 为 例 进行 说 明 ( 灰色 方 格 已 被 排除 了 )。 









































我 们 可 以 重复 套用 以 上 条 件 和 流程 找 出 55。 注 意 ， 在 此 只 能 使 用 条 件 1 和 条 件 4。 
下 面 是 这 个 排除 算法 的 实现 代码 。 








1 public static boolean findElement(int[][] matrix, int elem) { 
2 int row = ©; 

3 int col = matrix[6].length - 1; 

4 while (row < matrix.length && col >= 06) { 

5 if (matrix[row][col] == elem) { 

6 return true; 

7 } else if (matrix[row][col] > elem) { 

8 5 


COl--; 
9 } else { 
16 row++; 
11 } 
12 
13 return false; 
14 } 


还 有 别 的 做 法 ,我 们 可 以 运用 男 一 种 看 起 来 更 像 是 二 分 查找 法 的 解法 。 其 中 代码 要 复杂 得 多 ， 
但 也 用 到 了 很 多 相同 的 技巧 。 

解法 2: 二 分 查找 法 

让 我 们 再 来 看 个 简单 的 例子 。 


























15 | 20 | 7e | ss 
20 | 35 | se | 95 
30 | 55 | 95 | 105 
48 | 80 | lee | 120 
我 们 希望 能 够 充分 利用 矩阵 行列 已 排序 的 条 件 ， 以 更 有 效率 地 找到 元 素 。 因 此 , 试 着 问 问 自 
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己 ， 对 于 某 个 元 素 可 能 位 于 什么 位 置 ， 这 个 矩阵 独特 的 排序 属性 意味 着 什么 ? 
我 们 知道 每 一 行 每 一 列 都 是 已 排序 的 ， 也 就 是 说 元 素 a[i][j] 会 大 于 位 于 行 i、 列 0 和 列 j 一 1 
之 间 的 元 素 ， 并 且 大 于 位 于 列 j、 行 0 和 行 i- 1 之 间 的 元 素 。 
































换 句 话说 : 
a[li][e] <= a[li][1] <= ... <= a[li][j-1] <= a[i][j] 
a[le][j] <= a[l1][j] <= ... <= a[li-1][j] <= a[i][j] 


下 面 以 图 表 说 明 ， 其 中 深 灰 色 元 素 大 于 所 有 浅 灰色 元 素 。 





EE 


EE 105 


126 


浅 灰 色 元 素 也 有 顺序 : 每 一 个 都 大 于 它 左 边 的 元 素 , 并 且 大 于 它 上 方 的 元 素 , 因 此, 根据 传 
递 性 ， 深 灰色 元 素 比 色 块 里 的 其 他 元 素 都 要 大 。 



































这 意味 着 ， 若 在 矩阵 里 任意 画 个 长 方形 ， 其 右 下 角 的 元 素 一 定 是 最 大 的 。 
同样 地 ， 左 上 角 的 元 素 一 定 是 最 小 的 。 下 图 的 颜色 标示 元 素 的 大 小 顺序 〈 浅 灰色 < 深 灰色 < 
黑色 ): 


















20 | 76 | 85 
55 EE 
| 40 | 80 本 和 人 


让 我 们 回 到 原先 的 问题 : 假设 要 查找 值 83， 若 顺 着 对 角 线 搜索 ， 可 找到 元 素 33 和 95。 利 用 这 
些 信息 可 知 85 的 位 置 吗 ? 











1065 


| 40 | 86 120 120 


85 不 可 能 位 于 黑色 区 域 , 因为 95 位 于 该 区 域 的 左上 角 , 也 是 该 方形 里 最 小 的 元 素 。85 也 不 可 

















能 位 于 浅 灰 色 区 域 ， 因 为 35 位 于 该 方形 的 右 下 角 ， 是 该 方形 中 最 大 的 元 素 。 
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85 必 定位 于 两 个 白色 区 域 之 一 。 

因此 , 我 们 将 矩阵 分 为 四 个 区 域 ， 以 递归 方式 搜索 左下 区 域 和 右上 区 域 。 这 两 个 区 域 也 会 被 
分 成 子 区 域 并 继续 搜索 。 

注意 到 对 角 线 是 已 排序 的 ， 因 此 可 以 利用 二 分 查找 法 进行 高 效 的 搜索 。 

下 面 是 该 算法 的 实现 代码 。 


























1 public Coordinate findElement(int[][] matrix，Coordinate origin， 
2 Coordinate dest, int x) { 

3 if (lorigin.inbounds(matrix) || !dest.inbounds(matrix)) { 
4 return null; 

5 } 

6 if (matrix[origin.row][origin.column] == x) { 

7 return origin; 

8 } else if (!origin.isBefore(dest)) { 

9 return null; 

16 } 

11 

12 /* 将 start 和 end 分 别 设 为 对 角 线 的 起 点 和 终点 。 

13 # 和 矩阵 不 一 定 是 正方 形 ， 因 此 对 角 线 的 终点 也 

14 * 可 能 不 等 于 dest */ 

15 Coordinate start = (Coordinate) origin.clone(); 

16 int diagDist = Math.min(dest.row - origin.row, 

17 dest.column - origin.column); 

18 Coordinate end = new Coordinate(start.row + diagDist, 

19 start.column + diagDist); 
26 Coordinate p = new Coordinate(0, 0); 

21 


22 /* 在 对 角 线 上 进行 二 分 查找 ， 找 出 第 一 个 
23 * 比 X 大 的 元 素 */ 
24 while (start.isBefore(end)) { 


25 p.setToAverage(start, end); 

26 if (x > matrix[p.row][p.column]) { 
27 start.row = p.row + 1; 

28 start.column = p.column + 1; 

29 } else { 

36 end.row = p.row - 1; 

31 end.column = p.column - 1; 

32 } 

33 } 

34 


35  ”/* 将 矩阵 分 为 四 个 区 域 ， 搜 索 左 下 区 域 和 
36 * 右上 区 域 */ 


37 return partitionAndSearch(matrix, origin, dest, start, x); 
38 } 

39 

46 public Coordinate partitionAndSearch(int[][] matrix, 

41 Coordinate origin, Coordinate dest, Coordinate pivot, 

42 int elem) { 

43 Coordinate lowerLeftOrigin = 

44 new Coordinate(pivot.row, origin.column); 

45 Coordinate lowerLeftDest = 

46 new Coordinate(dest.row, pivot.column - 1); 
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47 Coordinate upperRightOrigin = 


48 new Coordinate(origin.row, pivot.column); 

49 Coordinate upperRightDest = 

56 new Coordinate(pivot.row - 1, dest.column); 

51 

52 Coordinate lowerLeft = 

53 findElement(matrix, lowerLeftOrigin, lowerLeftDest, elem); 
54 if (lowerLeft == null) { 

55 return findElement(matrix, upperRightOrigin, 

56 upperRightDest, elem); 

57 } 

58 return lowerLeft; 

59 } 

66 

61 public static Coordinate findElement(int[][] matrix, int x) { 
62 Coordinate origin = new Coordinate(06, 0); 

63 Coordinate dest = new Coordinate(matrix.length - 1， 

64 matrix[6].length - 1); 
65 return findElement(matrix, origin, dest, x); 

66 } 

67 

68 public class Coordinate implements Cloneable { 

69 public int row; 

76 public int column; 

71 public Coordinate(int r, int c) { 

了 之 row = r; 

73 column = Cj 

74 } 

75 

76 public boolean inbounds(int[][] matrix) { 

pd return row >= 6 && column >= 6 && 

78 row < matrix.length && column < matrix[6].length; 
79 } 

86 

81 public boolean isBefore(Coordinate p) { 

82 return row <= p.row && column <= p.column; 

83 } 

84 

85 public Object clone() { 

86 return new Coordinate(row, column); 

87 } 

88 

89 public void setToAverage(Coordinate min, Coordinate max) { 
96 row = (min.row + max.row) / 2; 

91 column = (min.column + max.column) / 2; 

92 } 

93 } 


如 果 你 读 过 上 面 所 有 代码 ， 心 里 会 想 :“ 我 可 没 办 法 在 面试 时 写 出 所 有 这 些 代 码 !” 没 错 ,的 
确 无 法 全 部 写 出 。 但 是 ,你 在 任何 面试 题 上 的 表现 都 会 比照 其 他 求职 者 进行 评估 , 因此， 如 果 你 


无 法 完整 写 出 代码 ， 他 们 也 同样 不 能 。 碰 到 这 类 棘手 的 问题 
将 一 些 代 码 独立 出 来 写成 方法 ， 可 以 增加 你 的 亮点 。 例 
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时 ， 你 未 必 处 于 不 利 的 位 置 。 
如 ， 将 partitionAndsearch 独 立 出 
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来 写成 一 个 方法 ， 想 勾勒 代码 的 轮廓 就 要 简单 许多 。 之 后 有 时 间 的 话 ， 你 可 以 再 回头 填充 
partitionAndsearch 的 内 容 。 


11.7 有 个 马戏 团 正在 设计 鞠 罗 汉 的 表演 节目 ， 一 个 人 要 站 在 另 一 人 的 肩膀 上 。 出 于 实际 和 
美观 的 考虑 ， 在 上 面 的 人 要 比 下 面 的 人 矮 一 点 、 轻 一 点 。 已 知 马 戏 团 每 个 人 的 高 度 和 重量 ,请 编 
写 代 码 计 算 苇 罗汉 最 多 能 对 几 个 人 。( 第 77 页 ) 

解法 

去 掉 此 题 的 “ 细 枝 末节 ”， 可 以 看 出 真正 要 考 的 题目 如 下 。 

给 定 一 个 列表 ,每 个 元 素 由 一 对 项 目 组 成 。 找 出 最 长 的 子 序列 , 其 中 第 一 项 和 第 二 项 均 以 非 
递减 的 顺序 排列 。 

如 果 套 用 简单 构造 法 〈 或 模式 匹配 法 )， 我 们 就 可 以 将 此 题 视 为 如 何 找 出 数组 中 的 最 长 递增 
序列 。 


1. 子 问题 : 最 长 递增 子 序列 
如 果 元 素 不 必 保 持 一 样 (相对 ) 的 顺序 ， 则 只 需 对 数组 进行 排序 即 可 。 这 么 一 来 ， 此 题 就 显 
得 太 过 简单 了 ， 因 此 ， 让 我 们 假设 元 素 必须 保持 一 样 的 相对 顺序 。 
通过 一 个 一 个 地 观察 数组 元 素 ， 可 以 试 着 推导 出 递归 算法 。 首 先 , 你 需要 了 解 ， 就算 知 道 了 
AL[8] 到 A[i] 的 最 长 递增 子 序列 ， 我 们 也 无 法 得 知 A[i + 1] 和 A[i + 2] 的 答案 。 这 一 点 由 下 面 这 
个 简单 的 例子 可 知 : 
数组 : 13，14，16，11，12 
Longest(6 through 6): 13 
Longest(6 through 1): 13, 14 
Longest(6 through 2): 13, 14 
Longest(8 through 3): 13, 14 或 16, 11 
Longest(6 through 4): 106, 11, 12 
如 果 只 是 试 着 以 最 新 的 解决 方案 求 出 Longest(@ through 4) 和 Longest(8 through 3)， 就 
会 找 不 到 最 优 解 。 
然而 ， 我 们 可 以 换 一 种 不 同 的 递归 解法 ,之 前 是 试 着 找 出 从 0 到 i 的 元 素 的 最 长 递增 子 序列 ， 
现在 改 为 找 出 以 元 素 i 结 尾 的 最 长 递增 子 序列 。 继 续 使 用 上 面 的 例子 ， 做 法 如 下 : 
数组 : 13，14，16，11，12 
Longest(ending with A[6]): 13 
Longest(ending with A[1]): 13, 14 
Longest(ending with A[2]): 16 
Longest(ending with A[3]): 106, 11 
Longest(ending with A[4]): 106, 11, 12 


注意 ， 以 A[i] 结 尾 的 最 长 子 序列 可 以 通过 检查 先前 全 部 解法 得 出 ， 只 要 将 A[i] 附 加 到 最 长 
且 “ 有 效 ” 的 那个 序列 即 可 ,所谓 “有 效 ” 是 指 符合 A[i] > list.tail 的 任意 序列 。 


2. 真正 的 问题 : 最 长 递增 子 序 列 ， 每 个 元 素 均 为 一 对 项 目 
现在 ,我 们 知道 如 何 找 出 一 串 整 数 的 最 长 递增 子 序 列 ， 就 可 以 很 容易 地 解决 真正 的 问题 ， 只 
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要 将 一 列表 演 人 员 按 身高 排序 ， 然 后 对 体重 套用 longestIncreasingsubsequence 算 法 即 可 。 
下 面 是 该 算法 的 实现 代码 。 








ArrayList<HtNt> getIncreasingSequence(ArrayList<HtWt> items) { 
Collections.sort(items); 
return longestIncreasingSubsequence(items); 


} 


1 
2 
3 
4 
5 
6 void longestIncreasingSubsequence(ArrayList<HtWt> array, 

7 ArrayList<HtNt>[] solutions, int current_ index) { 

8 if (current index >= array.size() || current index < 6) return; 
9 HtWt current element = array.get(current_ index); 

41 /* 找 出 可 以 附加 current_element 的 最 长 子 序列 */ 

12 ArrayList<HtWt> best_sequence = null; 


13 for (int i = 8; i < current_ index; i++) { 

14 if (array.get(i).isBefore(current element)) { 

15 best_sequence = seqWithMaxLength(best_sequence, 
16 solutions[i]); 

17 } 

18 } 

19 


26 /* 附加 current_element */ 
21 ArrayList<HtWt> new_solution = new ArrayList<Htwt>(); 
22 if (best_sequence != null) { 


23 new_solution.addAll(best_sequence); 

24 } 

25 new_solution.add(current_element) ; 

26 

27 /* 加 入 到 列表 中 ， 然 后 递归 */ 

28 solutions[current_index] = new_solution; 

29 longestIncreasingSubsequence(array, solutions, current index+1); 
36 } 

31 

32 ArrayList<HtWt> longestIncreasingSubsequence( 

33 ArrayList<HtNt> array) { 

34 ArrayList<HtWt>[] solutions = new ArrayList[array.size()]; 
35 longestIncreasingSubsequence(array, solutions, 0); 

36 

37 ArrayList<HtWt> best_sequence = null; 

38 for (int i = 6;j i < array.size(); i++) { 

39 best_sequence = seqWithMaxLength(best_ sequence, solutions[i]); 
40 } 

41 

42 return best_sequence; 

43 } 

44 


45 /* 返回 较 长 的 序列 */ 

46 ArrayList<Htwt> seqWithMaxLength(ArrayList<HtWt> seq1， 
47 ArrayList<HtWt> seq2) { 

48 if (seq1 == null) return seq2; 

49 if (seq2 == null) return seq1; 

56 return seq1.size() > seq2.size() ?seql : seq2; 
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53 public class HtWt implements Comparable { 
54 ”/* 声明 等 */ 


56 /* 供 sort 方 法 使 用 */ 
57 public int compareTo( Object s ) { 


58 HtWt second = (HtNt) s; 

59 if (this.Ht != second.Ht) { 

66 return ((Integer)this.Ht).compareTo(second.Ht); 
61 } else { 

62 return ((Integer)this.Wt).compareTo(second.Wt); 
63 } 

64 } 

65 

66 /* 若 this 应 该 排 在 other 之 前 ， 则 返回 true。 

67 * 注意 ，this.isBefore(other) 和 other.isBefore(this) 
68 * 两 者 避 为 false。 这 跟 compareTo 方 法 不 同 ， 

69 * 若 a < b, 则 b > a */ 

70 public boolean isBefore(HtWt other) { 

71 if (this.Ht < other.Ht && this.Wt < other.Wt) return true; 
了 2 else return false; 

73 } 

74 } 


这 个 算法 的 时 间 复 杂 度 为 0(*)， 确实 有 个 算法 的 用 时 可 以 达到 O(n log(n))， 但 要 复杂 得 多 ， 
不 太 可 能 在 面试 中 推导 出 来 ,哪怕 是 有 提示 也 办 不 到 。 不 过 ， 如 果 你 有 兴趣 试 试 这 种 解法 ,不妨 
上 网 搜 一 下 ， 应 该 能 搜 到 该 解法 的 不 少 说 明 。 


11.8 假设 你 正在 读 取 一 串 整数 。 每 隔 一 段 时 间 ， 你 希望 能 找 出 数字 * 的 秩 〈 小 于 或 等 于 x 的 
值 的 数目 )。 请 实现 数据 结构 和 算法 支持 这 些 操 作 。 也 就 是 说 ， 实 现 track(int x) 方 法 ， 每 读 入 
一 个 数字 都 会 调用 该 方法 ; 以 及 getRankOfNumber(int x) 方 法 ， 返 回 值 为 小 于 或 等 于 x 的 元 素 个 
数 (不 包括 x 本 身 )。( 第 77 页 ) 

解法 

有 种 相对 简单 的 实现 方式 是 用 一 个 数组 存放 所 有 已 排 好 序 的 元 素 。 当 有 新 元 素 进来 时 , 我 们 
需要 搬移 其 他 元 素 以 腾 出 空间 。 这 人 么 一 来 ，getRankOfNumper 实 现 起 来 会 非常 有 效 ， 只 需 执行 二 
分 查找 ， 返 回 索引 。 

然而 ， 插 和 元素 〈 也 就 是 track(int x) 函 数 ) 将 会 非常 低 效 ， 我 们 需要 一 种 数据 结构 ， 不 
仅 能 在 搬入 新 元 素 时 加 以 更 新 ， 还 能 维持 相对 排列 顺序 。 二 又 查找 树 正 好 适用 。 

之 前 是 要 把 元 素 插 入 数组 ， 现 在 则 要 将 元 素 插 入 二 又 查 找 树 。track(int x) 方 法 的 时 间 复 
杂 度 为 O(log n)， 其 中 n 为 树 的 大 小 ( 当然 ， 前 提 为 这 棵 树 是 平衡 的 )。 

要 找 出 某 个 数 的 秩 ， 可 以 执行 中 序 遍 历 , 并 在 访问 结 点 时 利用 计数 器 记录 数量 。 目 标 是 找到 
x 上 时， 计数 带 变量 将 会 是 小 于 x 的 元 素 的 数量 。 

在 查找 zx 期 间 ， 只 要 疝 左 移动 ， 计 数 需 变量 就 不 会 变 ， 为 什么 呢 ? 因为 右边 跳 过 的 所 有 值 都 
比 xz 大。 毕竟 最 小 的 元 素 〈 秩 为 1 ) 是 最 左边 的 结 点 。 
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可 是 当 向 右 移动 时 ， 我 们 跳 过 了 左边 的 一 堆 元 素 。 这 些 元 素 都 比 xz 小 ， 因 此 ， 必 须 增 加 计数 
顺 的 值 ， 这 个 值 等 于 左 子 树 的 元 素 个 数 。 

我 们 不 会 去 计算 左 子 树 的 大 小 (效率 低 )， 而 是 在 加 入 新 元 素 时 ， 记 录 相 关 信 息 。 

接 下 来 将 以 下 面 的 树 为 例 说 明 。 在 下 图 中 ,括号 内 的 数字 代表 左 子 树 的 结 点 数量 (或 者 , 换 
句 话 说， 该 结 点 相对 于 它 的 子 树 的 秩 )。 











假设 我 们 想 知道 34 在 上 面 这 棵 树 中 的 秩 , 会 先 将 24 与 根 结 点 20 比 较 , 发 现 24 位 于 右边 。 根 结 
点 的 左 子 权 有 4 个 结 点 ， 再 加 上 根 结 点 本 身 ， 总 共有 5 个 结 点 小 于 24， 因 此 我 们 会 将 计数 器 变量 
counter 设 为 5。 


然后 ， 将 24 与 结 点 25 进 行 比较 ， 发 现 24 必 定位 于 左边 。counter 变 量 的 值 不 会 更 新 ， 因 为 我 
们 并 未 “ 跳 过 ”任何 较 小 的 结 点 ，counter 变 量 的 值 仍 为 $。 

接着 , 将 24 与 结 点 23 进 行 比较 , 发现 24 必 定位 于 右边 。counter 变 量 会 增加 1 ( 变 为 6 )， 因 为 
23 没 有 左边 的 结 点 。 

最 后 ， 我 们 找到 24 并 返回 counter 值 : 6。 

这 个 递归 算法 如 下 : 








1 int getRank(Node node, int x) { 

2 if x is node.data 

3 return node.leftSize() 

4 if x is on left of node 

5 return getRank(node.left, x) 

6 if x is on right of node 

了 return node.leftSize() + 1 + getRank(node.right, x) 
8 


} 
下 面 是 完整 的 代码 。 





public class Question { 
private static RankNode root = null; 


1 

2 

3 

4 public static void track(int number) { 
5 if (root == null) { 

6 root = new RankNode(number); 

7 } else { 

8 root.insert(number); 

9 

1 


} 


© 


} 
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11 

12 public static int getRankOfNumber(int number) { 
13 return root.getRank(number); 
14 } 

15 

16 

17 } 

18 

19 public class RankNode { 

26 public int left_ size = 0; 

21 public RankNode left, right; 
22 public int data = 0@; 

23 public RankNode(int d) { 


24 data = d; 

25 } 

26 

27 public void insert(int d) { 

28 if (d <= data) { 

29 if (left != null) left.insert(d); 
36 else left = new RankNode(d); 

31 left_ sizet+; 

32 } else { 

33 if (right != null) right.insert(d); 
34 else right = new RankNode(d); 

35 } 

36 } 

37 

38 public int getRank(int d) { 

39 if (d == data) { 

46 return left_size; 

41 } else if (d < data) { 

42 if (left == null) return -1; 

43 else return left.getRank(d); 

44 } else { 

45 int right rank = right == null ? -1 : right.getRank(d); 
46 if (right_rank == -1) return -1; 

47 else return left size + 1 + right_rank; 
48 } 

49 } 

56 } 


注意 上 面 的 代码 是 怎么 处 理 4 不 在 树 里 的 情况 的 。 我 们 会 检查 返回 值 是 否 为 -1 ， 当 发 现 为 -1 
时 ， 将 它 往 上 返回 。 你 必须 处 理 诸如 此 类 情况 ， 这 很 重要 。 





9.12 测试 


12.1 找 出 以 下 代码 中 的 错误 〈 可 能 不 止 一 处 ): 


1 unsigned int i; 
2 for (i = 160; i >= 6; --i) 
3 printf(“%dNn”，i); (第 82 页 ) 
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解法 

这 段 代 码 有 两 处 错误 。 

首先 ， 根 据 定 义 ，unsigned int 类 型 的 变量 一 定 会 大 于 或 等 于 零 。 因 此 ，for 循 环 的 测试 条 
件 一 直 为 真 ， 将 陷入 无 限 循 环 。 

要 打印 100 到 1 之 间 的 所 有 整数 ， 正 确 的 做 法 是 测试 ?> 0。 如 果真 的 想 打印 0， 可 以 在 for 循 环 
之 后 加 一 条 printf 语 句 。 


1 unsigned int i; 
2 for (i = 160; i > 60; --i) 
3 printf(“%d\n”, i); 


另 一 个 需要 修正 的 地 方 是 用 Xu 代替 X%d， 因 为 这 里 打印 的 是 unsigned int 型 变量 。 


1 unsigned int i; 
2 for (i = 160; i > 6; --i) 
3 printf(“%u\n”, i); 


现在 ， 这 段 代 码 会 正确 地 打印 100 到 1 的 整数 序列 〈 按 降序 排列 )。 


12.2 ”有 个 应 用 程序 一 运行 就 出演， 现在 你 拿 到 了 源码 。 在 调试 器 中 运行 10 次 之 后 ， 你 发 
现 该 应 用 每 次 出 溃 的 位 置 都 不 一 样 。 这 个 应 用 只 有 一 个 线程 ， 并 且 只 调用 C 标 准 库 函 数 。 究 竟 是 
什么 样 的 编程 错误 导致 程序 崩溃 9 该 如 何 逐 一 测试 每 种 错误 ? (第 83 页 ) 


解法 

具体 如 何 处 理 这 个 问题 要 视 待 诊断 应 用 程序 的 类 型 而 定 。 不过, 我 们 还 是 可 以 给 出 一 些 随 机 
崩 演 的 常见 原因 。 

(1) 随机 变量 : 该 应 用 程序 可 能 用 到 某 个 随机 变量 或 可 变 分 量 ， 程 序 每 次 执行 时 取 值 不 定 。 
具体 的 例子 包括 用 户 输入 、 程 序 生成 的 随机 数 ， 或 当前 时 间 等 。 

(2) 未 初始 化 变量 : 该 应 用 程序 可 能 包含 一 个 未 初始 化 变量 ， 在 某 些 语言 中 ， 该 变量 可 能 含 
有 任意 值 。 这 个 变量 取 不 同 值 可 能 导致 代码 每 次 执行 路 径 有 所 不 同 。 

(3) 内 存 泄漏 : 该 程序 可 能 存在 内 存 溢出 。 每 次 运行 时 引发 问题 的 可 疑 进程 随机 不 定 ， 这 与 
当时 运行 的 进程 数量 有 关 。 男 外 还 包括 堆 溢 出 或 栈 内 数据 被 破坏 。 

(4) 外 部 依赖 : 该 程序 可 能 依赖 别 的 应 用 程序 、 机 器 或 资源 。 要 是 存在 多 处 依赖 ， 程 序 就 有 
可 能 在 任意 位 置 崩溃 。 

为 了 找 出 问题 的 原因 , 我 们 首先 应 该 尽 可 能 地 了 解 这 个 应 用 程序 。 谁 在 运行 这 个 程序 ? 他 们 
用 它 做 什么 ? 这 个 程序 属于 哪 种 应 用 ? 

此 外 ， 尽 管 应 用 程序 每 次 崩 演 的 位 置 不 尽 相 同 ,但 还 是 有 办 法 确定 它 可 能 与 特定 组 件 或 
场景 有 关 。 例 如 ， 有 可 能 只 是 启动 该 应 用 程序 而 不 进行 其 他 操作 时 ， 这 个 程序 从 不 朋 演 。 它 
只 有 在 载 人 文件 之 后 的 某 个 时 间 点 才 会 崩 演 。 或 者 ， 有 可 能 每 次 崩 淡 都 出 现在 底层 组 件 如 文 
件 IO 上 。 

要 解决 这 个 问题 ， 消 除法 也 许 值得 一 试 。 首 先 ,关闭 系统 中 其 他 所 有 应 用 ,仔细 追踪 资源 使 
用 。 如 果 该 程序 有 些 部 分 可 以 关 掉 ， 那 就 设法 关 掉 。 在 男 一 台 机 器 上 运行 该 程序 ， 看 看 能 否 重 现 
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同一 问题 。 我 们 可 以 消除 (或 修改 ) 的 越 多 ， 就 越 容易 定位 原因 。 

此 外 ,我 们 还 可 以 借助 工具 检查 特定 情况 。 例 如 ,要 排查 前 面 第 二 个 原因 ,我 们 可 以 利用 运 
行 时 工具 来 检查 未 初始 化 变量 。 

这 些 问题 不 仅 考查 你 解决 问题 的 方式 , 还 考查 你 头脑 风暴 的 能 力 。 你 是 否 会 像 热 锅 上 的 蚂蚁 ， 
胡乱 给 出 一 些 建议 ? 抑或 以 合乎 逻辑 的 、 有 条 理 的 方式 处 理 问题 ? 希望 是 后 者 。 


12.3 ”有 个 国际 象棋 游戏 程序 使 用 了 方法 : boolean canMoveTo(int x，int y)， 这 个 
方法 是 Piece 类 的 一 部 分 , 可 以 判断 某 个 棋子 能 否 移 动 到 位 置 (x, y)。 请 说 明 你 会 如 何 测试 该 方法 。 
(第 83 页 ) 

解法 

这 个 问题 主要 涉及 两 大 类 测试 : 极限 情况 测试 (确保 有 错误 输入 时 程序 不 会 骨 演 ) 和 一 般 情 
况 测试 。 我 们 先 从 第 一 类 测试 开始 。 


测试 类 型 1: 极限 情况 测试 

确保 程序 会 妥善 处 理 错误 或 异常 输入 ， 这 意味 着 要 检查 以 下 情况 : 

口 测试 x 和 y 为 负数 的 情况 ; 

口 测试 x 大 于 棋盘 宽度 的 情况 ; 

口 测试 y 大 于 棋盘 高 度 的 情况 ; 

口 测试 一 个 满 是 模子 的 棋盘 ; 
| 
| 


















































口 测试 一 个 空 或 接近 空 的 棋盘 ; 
口 测试 白 子 远 多 于 黑子 的 情况 ; 
口 测试 黑子 远 多 于 白 子 的 情况 。 
对 于 上 面 的 错误 情况 ,我 们 应 该 询问 面试 官 ,是 要 返回 false 还 是 抛 出 异常 ， 然 后 有 针对 性 地 
进行 测试 。 


























测试 类 型 2: 一 般 情况 测试 

一 般 情 况 测试 的 涉及 面 要 大 得 多 。 理 想 的 做 法 是 测试 每 一 种 可 能 的 棋盘 布局 , 但 是 棋局 实在 
太 多 了 。 不 过 ， 我 们 还 是 可 以 合理 地 执行 测试 ， 尽 量 涵盖 不 同 的 棋局 。 
国际 象棋 一 共有 6 种 棋子 ， 我 们 可 以 测试 每 一 种 棋子 ， 在 所 有 可 能 的 方向 上 ， 疝 其 他 所 有 棋 
子 移动 的 情况 。 大 致 如 下 面 的 代码 所 示 : 
























































1 对 每 一 种 棋子 a: 
2 ”对 其 他 每 一 种 棋子 D (6 种 及 空白 ) 
3 对 每 一 个 方向 d 
4 创建 有 a 的 棋盘 
5 将 b 放 在 方向 d 上 

试 着 移动 一 检查 返回 值 


此 题 的 关键 在 于 ， 认 识 到 我 们 不 可 能 测试 每 一 种 可 能 的 场景 ， 即 使 有 心 也 无 力 办 到 。 相 反 ， 
我 们 必须 专注 于 最 重要 的 部 分 。 
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它 还 


12.4 ”不 借助 任何 测试 工具 ， 该 如 何 对 网 页 进行 负载 测试 ? 〈 第 83 页 ) 

解法 

负载 测试 ( load test ) 不 仅 有 助 于 定位 Web 应 用 性 能 的 瓶 须 , 还 能 确定 其 最 大 连接 数 。 同样 地 ， 
能 检查 应 用 如 何 响应 各 种 负载 情况 。 

要 进行 负载 测试 ， 必须 先 确定 对 性 能 要 求 最 高 的 场景 ， 以 及 满足 目标 的 性 能 衡量 指标 。 一 般 





来 说 ， 有待 测 量 的 对 象 包括 : 


口 响应 时 间 ; 

口 否 吐 量 ; 

口 资源 利用 率 ; 

口 系统 所 能 承受 的 最 大 负载 。 

随后 ， 我 们 设计 各 种 测试 模拟 负载 ， 细 心 测量 上 面 的 每 一 项 。 
若 缺 少 正 规 的 测试 工具 ,我 们 可 以 自行 打造 。 例 如， 可 以 创建 成 千 上 万 的 虚拟 用 户 , 模拟 并 

















发 用 户 。 我 们 会 编写 多 线程 的 程序 ,新 建成 于 上 万 个 线程 ,每 个 线程 扮演 一 个 实际 用 户 , 载 入 待 
测 页 面 。 对 于 每 个 用 户 ， 可 以 利用 程序 来 测量 响应 时 间 、 数 据 IO (输入 /输出 )， 等 等 。 























之 后 ， 还 要 分 析 测 试 期 间 收集 的 数据 结果 ， 并 与 可 接受 的 值 进行 比较 。 
12.5 ”如 何 测试 一 支 笔 (第 83 页 ) 


解法 
这 个 问题 很 大 程度 上 在 于 理解 限制 条 件 ， 并 有 条 理 、 结 构 化 地 解决 该 问题 。 
为 了 理解 有 哪些 限制 条 件 ， 你 应 该 抛 出 一 系列 疑问 ， 针 对 某 个 问题 了 解 “ 谁 、 什 么 、 何 地 、 

















何 时 、 如 何以 及 为 什么 ”( 只 要 与 该 问题 相关 , 越 多 越 好 ), 一 个 好 的 测试 人 员 会 在 着 手 测 试 之 前 ， 
先 准 确 了 解 自 己 要 测试 的 是 什么 。 


为 了 说 明 上 面 这 项 技巧 ， 我 们 来 看 看 下 面 的 模拟 对 话 。 

面试 官 : 你 会 如 何 测试 一 支 笔 ? 

求职 者 : 我 想 先 了 解 一 下 这 支 笔 。 谁 会 使 用 这 支 笔 ? 

面试 官 : 可 能 是 小 孩 。 

求职 者 : 喝 ， 有 意思 。 他 们 会 用 这 文笔 做 什么 ?” 写字 、 画 画 还 是 干 别 的 ? 
面试 官 : 画 画 。 

求职 者 : 好 的 ， 谢 谢 。 画 在 哪里 呢 ? 纸张 、 布 料 还 是 墙壁 上 ? 
面试 官 : 画 在 布料 上 。 
求职 者 : 那么 , 这 文笔 的 笔头 是 什么 样 的 ? 签字 笔 还 是 圆珠笔 ? 要 洗 得 掉 的 ， 还 是 洗 不 掉 的 ? 
面试 官 : 要 求 洗 得 掉 。 

在 问 了 很 多 问题 之 后 ， 你 可 以 得 出 如 下 结论 。 

求职 者 : 好 的 ， 综 上 ， 我 理解 如 下 : 这 文笔 主要 面向 5 一 10 岁 的 小 孩 ， 为 签字 笔头 ， 有 红 、 



























































、 蓝 、 黑 四 色 ， 用 来 画 画 。 画 在 布料 上 并 且 要 求 洗 得 掩 。 我 的 理解 对 吗 ? 
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此 时 , 求职 者 面 对 的 问题 与 乍 看 上 去 的 问题 差异 很 大 ,这 种 情况 并 不 少见 。 实 际 上 , 许多 面 
试 官 会 故意 给 一 个 看 似 再 清楚 不 过 的 问题 ( 谁 不 知道 笔 是 什么 呢 ! )， 其 实 是 在 考查 你 , 看 你 能 否 
发 现 这 个 问题 与 最 初 理解 的 有 很 大 差别 。 他 们 相信 用 户 也 会 这 么 做 ,但 用 户 多 半 是 无 意 的 。 

至 此 ， 你 已 经 知道 自己 要 测试 的 是 什么 ， 接 下 来 该 提出 测试 计划 了 。 这 里 的 关键 是 结构 。 

想 想 测试 对 象 或 问题 会 涉及 哪些 方面 ,并 以 此 为 基础 展开 测试 。 这 个 问题 涉及 以 下 几 个 方面 。 
口 事实 核查 : 核实 这 是 一 支 签字 笔 ， 墨 水 颜色 为 要 求 的 四 种 颜色 之 一 。 

口 预期 用 途 : 绘制 ， 这 支 笔 在 布料 上 夯 得 出 来 吗 ? 

口 预期 用 途 : 水 洗 ， 夯 在 布料 上 的 墨迹 洗 得 掉 吗 ( 哪怕 已 经 过 了 一 段 时 间 ) ? 是 用 热 水 、 

温水 还 是 冷水 才能 洗 掉 ? 

口 安全 性 : 这 文笔 对 小 孩 是 否 安全 (无 毒 )? 

口 非 预期 用 途 : 小 孩 还 会 怎么 使 用 这 支 笔 ? 他 们 可 能 在 其 他 物体 表面 上 涂鸦 ， 因 此 还 需 检 
查 他 们 的 行为 是 否 正确 。 他 们 还 可 能 踩踏 、 乱 扔 这 支 笔 ， 等 等 。 你 需要 确认 这 支 笔 是 否 
经 受 得 住 这 些 使 用 条 件 。 

记 住 ， 对 于 任何 测试 问题 ， 你 都 必须 测试 预期 和 非 预 期 的 场景 。 人 们 并 不 一 定 按照 你 预想 的 
方式 使 用 产品 。 

12.6 ”在 一 个 分 布 式 银行 系统 中 ， 该 如 何 测试 一 人 台 ATM 机 ? (第 83 页 ) 

解法 

对 于 这 个 问题 ， 第 一 要 务 是 厘清 若干 假设 条 件 ， 请 提出 以 下 问题 。 

口 谁 会 使 用 ATM 机 ? 答案 可 能 是 “任何 人 ”， 或 是 “盲人 ”， 或 任意 其 他 可 能 的 答案 。 

口 他 们 会 用 ATM 机 来 做 什么 ?答案 可 能 是 “取款 ”"、“ 转 账 *"、“ 查 询 余额 "， 等 等 。 

口 我 们 有 什么 工具 来 测试 呢 ? 我 们 可 以 查看 代码 吗 ? 还 是 只 能 访问 ATM 机 ? 

记 住 ， 好 的 测试 人 员 会 先 确定 自己 要 测试 的 是 什么 。 

一 旦 了 解 系统 是 什么 样 的 ， 我 们 就 会 想 着 将 问题 分 解 成 可 测试 的 子 部 分 ， 包括: 

口 登录 ; 

口 取款 ; 

口 存款 ; 

口 查询 余额; 

口 转账 。 

我 们 可 能 要 搭配 使 用 手动 和 自动 测试 。 

手动 测试 会 检查 上 述 步骤 的 每 一 个 环节 ， 确 保 涵盖 所 有 错误 情况 (余额 不 足 、 新 开 账 户 、 不 
存在 的 账户 ， 等 等 )。 

自动 测试 稍微 复杂 一 点 。 我 们 会 希望 自动 处 理 上 述 所 有 标准 流程 , 还 要 找 一 些 非常 具体 的 问 
题 ， 比 如 竞争 条 件 。 理 想 情况 下 ,我 们 会 设法 建立 一 套 有 假 帐 户 的 封闭 系统 ， 以 确保 即使 有 人 从 
不 同 地 点 快速 取款 和 存款 ， 他 也 不 会 多 得 不 应 得 的 钱 ， 或 者 损失 应 得 的 钱 。 

最 重要 的 是 , 我 们 必须 优先 考虑 安全 性 和 可 靠 性 。 客 户 的 帐户 无 时 无 刻 都 要 处 于 被 保护 的 状 
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态 , 我 们 必须 确保 账目 得 到 正确 处 理 。 没 有 人 和 希望 自己 的 钱 不 波 而 飞 。 优秀 的 测试 人 员 深 说 整个 
系统 里 哪些 事项 是 最 重要 的 。 





9.13 C 和 C++ 


13.1 用 C++ 写 个 方法 ， 打 印 输入 文件 的 最 后 K 行 。( 第 88 页 ) 

解法 

此 题 有 一 种 蛮 力 法 : 先 数 出 文件 的 行 数 (和)， 然 后 打印 第 N-K 行 到 第 N 行 。 但 是 ， 这 么 做 ， 
文件 要 读 两 遍 ， 会 产生 没 必要 的 开销 。 我 们 需要 一 种 解法 ， 只 读 一 遍 文 件 就 能 打 印 最 后 K 行 。 

我 们 可 以 使 用 一 个 数组 ， 存 放 从 文件 读 取 到 的 所 有 K 行 和 最 后 的 K 行 。 因 此 ， 这 个 数组 起 初 
包含 的 是 0 ~K 行 ， 然 后 是 1 ~ K+1 行 ， 接 着 是 2 ~ K+2 行 ， 依 此 类 推 。 每 次 读 取 新 的 一 行 ， 就 将 数 
组 中 最 早 读 入 的 那 一 行 清 掉 。 

不 过 ,你 可 能 会 问 ,， 这么 做 是 不 是 还 要 移动 数组 元 素 ， 进而 引入 很 大 的 开销 ? 不 会 ， 只 要 做 
法 得 当 就 不 会 。 我 们 将 使 用 循环 式 数组 ， i 

使 用 循环 式 数 组 ( circular array )， 每 次 读 取 新 的 一 行 ， 都 会 替换 数组 中 最 早 读 人 的 元 素 。 我 
们 会 以 专门 的 变量 记录 这 个 元 素 ; en 

下 面 是 循环 式 数 组 的 例子 : 


步骤 1 (初始 态 ) : array = {a, b, c,d 
步骤 2 (插入 g) : array = {8g, b，c，d，e， f}. 
步骤 3 (插入 h) : array = {g，h，c，d 
步骤 4 (插入 i) : array = {g，h，i，d 


下 面 是 该 算法 的 实现 代码 。 


1 void printLast16Lines(chark fileName) { 
const int K = 108; 
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3 ifstream file (fileName); 

4 string L[K]; 

5 int size = 0) 

6 

7 /* 逐 行 读 取 文 件 ， 并 存 入 循环 式 数 组 */ 

8 while (file.good()) { 

9 getline(file, L[size % K]); 

16 Size++; 

了 } 

12 

13 /* 计算 循环 式 数组 的 开头 和 大 小 */ 

14 int start = size >K ?(Size % K) : 6 
15 int count = min(K, size); 

16 

17 /* 根据 读 取 顺序 ， 打印 数组 元 素 */ 

18 for (int i = 6; i «< count; i++) { 

19 cout << L[(start + i) % K] << endl; 
26 } 

21 } 
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这 种 解法 要 求 读 取 整 个 文件 ， 不 过 ， 任 意 时 刻 都 只 会 在 内 存 里 存放 10 行 内 容 。 


13.2 比较 并 对 比 散 列表 和 STL map。 散 列表 是 如 何 实现 的 ? 如 果 输 入 的 数据 量 不 大 ， 可 以 
选用 哪些 数据 结构 替代 散 列 表 ? (第 89 页) 


解法 

在 散 列表 里 , 值 的 存放 是 通过 将 键 传 入 散 列 函数 实现 的 。 值 并 不 是 以 排序 后 的 顺序 存放 。 此 
外 ， 散 列表 以 键 找 出 索引 ， 进 而 找到 存放 值 的 地 方 ， 因 此 ， 插 入 或 查找 操作 均 挫 后 可 以 在 O(1) 
时 间 内 完成 假定 该 散 列 表 很 少 发 生 碰撞 冲突 )。 散 列表 还 必须 处 理 潜在 的 碰撞 冲突 ， 一 般 通过 
拉链 法 (chaining ) 解决 ， 也 即 创 建 一 个 链表 来 存放 值 ， 这 些 值 的 键 都 映射 到 同一 个 索引 。 

STL map 的 做 法 是 根据 键 ， 将 键 值 对 插入 二 又 查找 树 。 不 需要 处 理 冲 突 ， 因 为 树 是 平衡 的 ， 
插入 和 查找 操作 的 时 间 肯 定 为 O(log N)。 


散 列表 是 如 何 实现 的 ? 

传统 上 ， 散 列表 都 是 用 元 素 为 链表 的 数组 实现 的 。 想 要 搬入 键 值 对 时 ， 先 用 散 列 函数 将 键 映 
射 为 数组 索引 ， 随 后 ， 将 值 搬入 那 个 索引 位 置 对 应 的 链表 。 

注意 ,在 数组 的 特定 索引 位 置 的 链表 中 ， 各 个 元 素 的 键 并 不 相同 ， 这 些 值 的 
hashFunction(key) 才 是 相同 的 。 因 此 , 为 了 取 回 某 个 键 对 应 的 值 , 每 个 结 点 都 必须 存放 键 和 值 。 

总 而 言 之 , 散 列表 会 以 链表 数组 的 形式 实现 , 链表 中 每 个 结 点 都 会 存放 两 块 数据 : 值 和 原先 
的 键 。 此 外 ， 我 们 还 要 注意 以 下 设计 准则 。 

(1) 我 们 希望 使 用 一 个 优良 的 散 列 函数 ， 确 保 能 将 键 均匀 分 散 开 来 。 奉 分 散 不 均匀 ， 就 会 发 
生 大 量 碰撞 冲突 ， 查 找 元 素 的 速度 也 会 变 慢 。 

(2) 不 论 散 列 函 数 选 的 多 好 ， 还 是 会 出 现 碰撞 冲突 ， 因 此 需要 一 种 碰撞 处 理 方法 。 通 常 ， 我 
们 会 采用 拉链 法 ， 也 就 是 通过 链表 来 处 理 ， 但 这 并 不 是 唯一 的 做 法 。 

(3) 我 们 可 能 还 希望 设法 根据 容量 动态 扩大 或 缩小 散 列 表 的 大 小 。 例 如 ， 当 元 素数 量 和 散 列 
表 大 小 之 比 超过 一 定 阐 值 时 ,可 能 会 希望 扩大 散 列 表 的 大 小 。 这 意味 着 要 新 建 一 个 散 列表 , 并 将 
旧 的 散 列表 条 目 转移 到 新 的 散 列表 中 。 因 为 这 种 操作 的 开销 非常 大 ， 所 以 我 们 要 谨慎 些 , 切 不 可 
频繁 操作 。 


如 果 输 入 的 数据 量 不 大 ， 可 以 选用 哪些 数据 结构 替代 散 列 表 ? 
你 可 以 使 用 STL map 或 二 义 树 。 尽 管 两 者 的 插入 操作 需要 O(log(n)) 的 时 间 ， 但 若是 输入 数据 
量 够 小 ， 这 点 时 间 就 可 以 忽略 不 计 。 


13.3 ”C++ 庶 函 数 的 工作 原理 是 什么 ” (第 89 页 ) 


解法 

虚 函 数 (virtual function ) 需要 虚 函 数 表 (vtable，Virtual Table ) 才能 实现 。 如 果 一 个 类 有 函 
数 声 明成 虚拟 的 ， 就 会 生成 一 个 vtable， 存 放 这 个 类 的 虚 函 数 地 址 。 此 外 ， 编 译 器 还 会 在 类 里 加 
入 隐藏 的 vptr 变 量 。 若 子 类 没有 覆 写 虚 函 数 ， 该 子 类 的 vtable 就 会 存放 父 类 的 函数 地 址 。 调 用 这 
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个 虚 函 数 时 ， 就 会 通过 vtable 解 析 函 数 的 地 址 。 在 C++ 里 ,动态 绑 定 ( dynamic binding ) 就 是 通过 
vtable 机 制 实现 的 。 

由 此 ， 将 子 类 对 象 赋值 给 基 类 指针 时 ，vptr 变 量 就 会 指向 子 类 的 vtable。 这 样 一 来 ， 就 能 确 
保 继 承 关系 最 末端 的 子 类 虚 函 数 会 被 调用 到 。 


请 考虑 以 下 代码 。 

1 class Shape { 

2 public: 

3 int edge_length; 

4 virtual int circumference () { 

5 cout << “Circumference of Base Class\n”; 
6 return 0@; 

7 } 

8 ); 

9 class Triangle: public Shape { 

16 public: 

11 int circumference () { 

12 cout<< “Circumference of Triangle Class\n”; 
13 return 3 * edge_length; 

14 } 

15 )}; 


16 void main() { 
17 Shape * x = new Shape(); 


18 x->circumference(); // “Circumference of Base Class” 

19 Shape *y = new Triangle(); 

26 y->circumference(); // “Circumference of Triangle Class” 
21: 


在 上 面 的 代码 中 ，circumference 是 Shape 类 的 虚 函 数 ， 因 此 在 所 有 继承 Shape 类 的 子 类 
(Triangle 等 ) 里 都 为 虚 函 数 。 在 C++ 里 ， 非 虚 函 数 的 调用 是 在 编译 期 通过 静态 绑 定 确定 的 ， 而 
虚 函 数 的 调用 则 是 在 运行 期 通过 动态 绑 定 确定 的 。 


13.4” 深 拷贝 和 浅 拷贝 之 间 有 何 区 别 9 请 说 明 两 者 的 用 法 。( 第 89 页 ) 





























解法 
浅 拷贝 会 将 对 象 所 有 成 员 的 值 拷贝 到 男 一 个 对 象 里 。 除了 拷贝 所 有 成 员 的 值 , 深 拷贝 还 会 进 
一 步 拷贝 所 有 指针 对 象 。 








下 面 是 浅 拷贝 和 深 找 贝 的 例子 。 


struct Test { 
char * ptr; 


}; 


dest.ptr = src.ptr; 


} 


void deep copy(Test & src, Test & dest) { 


1 

2 

3 

4 

5 void shallow copy(Test & src, Test & dest) { 

6 

7 

8 

9 

16 dest.ptr = (char *)malloc(strlen(src.ptr) + 1); 
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11 strcpy(dest.ptr, src.ptr); 
12 } 


注意 ，shallow_copy 可 能 会 导致 大 量 编程 运行 时 错误 ， 尤 其 是 在 对 象 创 建 和 销毁 时 。 使 
浅 找 贝 时 ,必须 非常 小 心 ， 只 有 当 开 发 人 员 真 正 知 道 自己 在 做 些 什么 时 方 可 选用 浅 拷贝 。 多 数 





























用 


情 


况 下 , 使 用 浅 拷贝 是 为 了 传递 一 块 复杂 结构 的 信息 , 但 又 不 想 真 的 复制 一 份 数据 。 使 用 浅 拷贝 时 ， 





销毁 对 象 必须 非常 小 心 。 


在 实际 开发 中 , 浅 拷贝 很 少 使 用 。 大 部 分 情况 都 应 该 使 用 深 拷贝 , 特别 是 当 需 要 拷贝 的 结 
很 小 时 。 

13.5”C 语 言 的 关键 字 “volatile” 有 何 作用 ? (第 89 页 ) 

解法 





关键 字 volatile 的 作用 是 指示 编译 器 ， 即 使 代码 不 对 变量 做 任何 改动 ， 该 变量 的 值 仍 可 能 














会 被 外 界 修改 。 操 作 系统 、 硬 件 或 其 他 线程 都 有 可 能 修改 该 变量 。 该 变量 的 值 有 可 能 遭受 意料 
外 的 修改 ， 因 此 ， 每 一 次 使 用 时 ， 编 译 器 都 会 重新 从 内 存 中 获取 这 个 值 。 
volatile ( 易 变 ) 的 整数 可 由 下 面 的 语句 声明 : 


int volatile x; 
volatile int x; 


要 声明 指向 volatile 整 数 的 指针 ， 可 以 这 么 做 : 


volatile int * x; 
int volatile * x; 


指向 非 volatile 数 据 的 volatile 指 针 很 少见 ,但 也 是 可 行 的 : 

int * volatile x; 

如 车 声明 指向 一 块 volatile 内 存 的 volatile 指 针 变 量 ( 指针 本 身 与 地 址 所 指 的 内 存 都 
volatile )， 做 法 如 下 : 

int volatile * volatile x; 


volatile 变 量 不 会 被 优化 掉 ， 这 非常 有 用 。 设 想 有 下 面 这 个 函数 : 





























1 int opt = 1; 

2 void Fn(void) { 

3 start: 

4 if (opt == 1) goto start; 
5 else break; 

6 


} 
乍 一 看 ， 上 面 的 代码 好 像 会 进入 无 限 循 环 ， 编 译 絮 可 能 会 将 这 段 代码 优化 成 : 





1 void Fn(void) { 

2 start: 

3 int opt = 1; 
4 if (true) 

5 goto start; 
6 


} 
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地 


那么 


构 函 


这 样 就 变 成 了 无 限 循环 。 然 后 ， 外 部 操作 可 能 会 将 0 瑟 入 变量 opt 的 位 置 ， 从 而 终止 循环 。 
为 了 防止 编译 器 执行 这 类 优化 , 我 们 需要 设法 通知 编译 需 , 系统 其 他 部 分 可 能 会 修改 这 个 变 
具体 做 法 就 是 使 用 volatile 关 键 字 ， 如 下 所 示 。 








1 Vvolatile int opt = 1; 

2 void Fn(void) { 

3 start: 

4 if (opt == 1) goto start; 
5 else break; 

6 } 





volatile 变 量 在 多 线程 程序 里 也 很 有 用 , 对 于 全 局 变量 , 任意 线程 都 可 能 修改 这 些 共享 的 变 
我 们 可 能 不 希望 编译 需 对 这 些 变量 进行 优化 。 

13.6” 基 类 的 析 构 函数 为 何 要 声明 为 virtual9 (第 89 页 ) 

解法 

让 我 们 先 想 想 为 何 会 有 虚 函 数 ， 假 设 有 如 下 代码 : 

















class Foo { 
public: 
void f(); 


于 

2 

3 

4 ); 
3 

6 class Bar : public Foo { 
2 
8 


public: 
void f(); 
9 } 
16 
11 Foo * p = new Bar(); 
12 p->f(); 




















调用 p->f() 最 后 将 会 调用 Foo: :f() ， 这 是 因为 p 是 指向 Foo 的 指针 ， 而 f() 不 是 虚拟 的 。 

为 确保 p->f() 会 调用 继承 关系 最 未 端的 子 类 的 f() 实 现 ， 我 们 需要 将 f() 声 明 为 虚 函 数 。 
现在 ， 回 到 前 面 的 析 构 函数 。 析 构 函 数 用 于 释放 内 存 和 资源 。Foo 的 析 构 函数 若 不 是 虚拟 的 ， 
， 即 使 p 实 际 上 是 Bar 类 型 的 ， 还 是 会 调用 Foo 的 析 构 函数 。 

这 就 是 为 何 要 将 析 构 函数 声明 为 虚拟 的 原因 一 一 确保 正确 调用 继承 关系 最 末端 的 子 类 的 析 
数 。 


13.7 编写 方法 ， 传 入 参数 为 指向 Node 结 构 的 指针 ， 返 回 传 入 数据 结构 的 完整 拷贝 。 其 中 ， 























Node 数 据 结构 含有 两 个 指向 其 他 Node 的 指针 。( 第 89 页 ) 


解法 
下 面 的 算法 将 记录 一 份 映 射 关 系 , 从 原先 结构 中 的 结 点 地 址 对 应 到 新 结构 中 相应 的 结 点 。 利 





用 该 映射 关系 ， 在 这 个 结构 的 深度 优先 遍历 中 ,就 能 判断 某 个 结 点 是 不 是 复制 过 了 。 遍历 时 通常 
会 标记 访问 过 的 结 点 ， 标 记 可 以 有 多 种 形式 ， 不 一 定 要 存放 在 结 点 里 。 


综 上 ， 可 以 得 到 一 个 简单 的 递归 算法 : 
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1 typedef map<Node*, Node*> NodeMap; 

2 

3 Node * copy_recursive(Node * cur, NodeMap & nodeMap) { 
4 if(cur == NULL) { 

5 return NULL; 

6 } 

7 

8 NodeMap::iterator i = nodeMap.find(cur); 
9 if (i != nodeMap.end()) { 

16 // 已 访问 过 这 里 ， 返 回 拷贝 

11 return i->second; 

12 } 

13 

14 Node * node = new Node; 


16 node->ptr1 = copy_recursive(cur->ptr1, nodeMap); 
17 node->ptr2 = copy_recursive(cur->ptr2, nodeMap); 
18 return node; 

19 } 


15 nodeMap[cur] = node; // 在 遍历 链接 之 前 ,建立 映射 关系 


21 Node * copy_structure(Node * root) { 
22 NodeMap nodeMap; // 需要 一 个 空 的 map 
23 return copy_recursive(root, nodeMap); 


13.8 ”编写 一 个 智能 指针 类 。 智 能 指针 是 一 种 数据 类 型 ， 一 般 用 模板 实现 ， 模 拟 指针 行为 
的 同时 还 提供 自动 垃圾 回收 机 制 。 它 会 自动 记录 smartPointer<T*> 对 象 的 引用 计数 ,一旦 T 类 型 
对 象 的 引用 计数 为 零 ， 就 会 释放 该 对 象 。( 第 89 页) 


解法 

智能 指针 跟 普 通 指针 一 样 ， 但 它 借 由 自动 化 内 存 管理 保证 了 安全 性 ， 避 免 了 诸如 悬挂 指针 、 
内 存 泄漏 和 分 配 失败 等 问题 。 智 能 指针 必须 为 给 定 对 象 的 所 有 引用 维护 单一 引用 计数 。 

第 一 次 看 到 这 类 问题 ， 可 能 会 觉得 太 难 而 不 知 所 措 ， 特 别 是 当 你 并 非 C++ 专 家 时 。 此 题 有 个 
解决 之 道 ， 分 两 步 走 :(1) 以 伪 码 勾勒 出 做 法 ; (2) 实现 具体 代码 。 

按照 这 种 做 法 ,我 们 需要 一 个 引用 计数 变量 ,每 新 增 一 个 对 象 的 引用 ,该 变量 会 加 一 , 移 除 
一 个 引用 则 减 一 。 实 现代 码 与 下 面 的 伪 码 类 似 : 





template <class T> class SmartPointer { 
/* 智能 指针 类 需要 指向 对 象 本 身 及 引用 计数 两 者 
* 的 指针 。 这 些 都 必须 是 指针 ， 而 不 是 真实 的 对 象 
* 或 引用 计数 值 ， 因 为 智能 指针 的 目的 就 在 于 ， 
可 以 跨 多 个 指向 某 一 对 象 的 智能 指针 ， 来 追踪 
同一 个 引用 计数 */ 
T * obj; 
unsigned * ref_count; 


} 
这 个 类 还 需要 疹 干 构造 函数 和 一 个 析 构 函数 ， 下 面 完 加 上 这 些 函 数 。 


1 SmartPointer(T * object) { 


关 





* 


1 
2 
3 
4 
5 
6 
7 
8 
9 
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2 /* 想 要 设 定 T * 0bj 的 值 ， 并 将 引用 计数 
3 * 设 为 1 */ 
4 ) 
5 
6 SmartPointer(SmartPointer<T>& sptr) { 
2 /* 这 个 构造 函数 会 新 建 一 个 指向 已 有 对 象 的 
8 * 智能 指针 。 我 们 需要 先 设 定 0bj 和 ref_count， 
9 *# 设 为 指向 sptr 的 obj 和 ref_count。 然 后 ， 
16 * 因为 我 们 新 建 了 一 个 obj 的 引用 ， 所 以 需要 
11 * 增加 ref_count */ 
12 } 
13 
14 ~SmartPointer(SmartPointer<T> sptr) { 
15 /* 销毁 该 对 象 的 引用 ,减少 ref_count 的 值 。 
16 * 车 ref_count 为 9， 则 释放 为 存放 整数 而 申请 的 内 存 ， 
17 * 并 销毁 对 象 */ 
18 } 


还 有 一 种 方式 也 可 以 创建 引用 : 将 一 个 smartPointer 赋 值 给 另 一 个 。 处 理 这 种 情况 需要 和 履 
写 = 操 作 符 ， 不 过 这 里 先 略 述 一 二 。 


19 onSetEquals(SmartPointer<T> ptr1，SmartPointer<T> ptr2) { 


20 
发 和 
22 
23 
24 


} 


/* 若 ptr1 已 有 值 ， 减 小 其 引用 计数 。 然 后 ， 
* 复制 指向 obj 和 ref_count 的 指针 。 最 后 ， 
* 因为 创建 了 新 引用 ， 所 以 需要 增加 

# ref_count 的 值 */ 





即使 尚未 填 和 人 复杂 的 C++ 语法 ， 仅 仅 把 做 法 大 致 描绘 出 来 ， 意 义 已 经 很 重大 了 。 接 下 来 ,要 
完成 所 有 代码 ， 只 需 填 补 好 细节 即 可 。 


1 
2 
3 
4 
5 
6 
2 
8 
9 


16 
11 
12 
13 
14 
15 
16 
7 
18 
19 
20 
21 
之 六 
23 


template <class T> class SmartPointer { 


public: 


SmartPpointer(T * ptr) { 
ref = ptr; 
ref_count = (unsigned*)malloc(sizeof(unsigned)); 
*ref_count = 1; 


} 


SmartPointer(SmartPointer<T> & sptr) { 
ref = sptr.ref; 
ref_count = sptr.ref_count; 
++(*ref_count); 


lj 


/* 惟 写 = 运算 符 ， 这样 才 能 将 一 个 旧 的 
* 智能 指针 赋值 给 另 一 指针 ， 旧 的 引用 
* 计数 减 一 ， 新 的 智能 指针 的 引用 计数 
*# 则 加 一 */ 
SmartPointer<T> & operator=(SmartPointer<T> & sptr) { 
if (this == &sptr) return *this; 


/* 车 已 赋值 为 某 个 对 象 ， 则 移 除 引 用 */ 
if (*ref_count > 9) { 
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24 remove(); 

25 } 

26 

27 ref = sptr.ref; 

28 ref_count = sptr.ref_count; 
29 ++(*ref_count); 

36 return *this; 

31 } 

32 

33 ~SmartPointer() { 

34 remove(); // 移 除 一 个 对 象 引用 
35 } 

36 

37 T getValue() { 

38 return *ref; 

39 } 

40 

41 protected: 

42 void remove() { 

43 --(*ref_count); 

44 if (*ref_count == 6) { 
45 delete ref; 

46 free(ref_count); 
47 ref = NULL; 

48 ref_count = NULL; 
49 } 

56 } 

5 

52 T * ref; 

53 unsigned * ref_count; 
54 }; 


此 题 的 代码 复杂 难 履 ， 错 漏 在 所 难免 ,面试 官 也 不 会 强求 代码 写 得 完美 无 缺 。 


13.9 ”编写 支持 对 齐 分 配 的 malloc 和 free 函 数 ， 分 配 内 存 时 ，malloc 函 数 返回 的 地 址 必须 能 
被 2 的 "次 方 整除 。( 第 89 页 ) 


解法 

一 般 来 说 , 使 用 malloc, 我 们 控制 不 了 分 配 的 内 存 会 在 堆 里 哪个 位 置 。 我 们 只 会 得 到 一 个 指 
向 内 存 块 的 指针 ， 指 针 的 起 始 地 址 不 定 。 

要 克服 这 些 限 制 条 件 , 我 们 必须 申请 足够 大 的 内 存 , 要 大 到 可 以 返回 可 被 指定 数值 整除 的 内 
存 地 址 。 

假设 需要 一 个 100 字 节 的 内 存 块 ， 我 们 希望 它 的 起 始 地 址 为 16 的 倍数 。 需 要 额外 分 配 多 少 内 
存 才 够 用 呢 ? 我 们 需要 额外 分 配 15 字 节 。 有 了 这 15 字 节 ， 加 上 紧 随 其 后 的 100 字 节 ， 就 能 得 到 可 
被 16 整 除 的 内 存 地 址 ， 以 及 100 字 节 的 可 用 空间 。 




















具体 做 法 大 致 如 下 : 
1 void* aligned malloc(size t required bytes, size t alignment) { 
之 int offset = alignment - 1; 
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3 void* p = (void*) malloc(required bytes + offset ) ; 

4 void* q = (void*) (((size t)(p) + offset) & ~(alignment - 1)); 

5 return q; 

6 } 

第 4 行 有 点 难 懂 ， 人 解释 如 下 。 假 设 alignment 为 16。 很 显然 ， 在 前 16 字 节 的 某 个 位 置 ， 肯 定 











有 个 内 存 地 址 可 被 16 整 除 。 执 行 (p1 + 16) & 11. .166686， 可 将 q 往 后 移 到 可 被 16 整 除 的 内 存 地 
址 。 并 地 址 未 4 位 和 8666 执 行 位 与 操作 ， 以 确保 新 的 值 可 被 16 整 除 。 

这 种 解法 近乎 无 可 挑剔 ， 只 是 有 个 大 问题 ， 如 何 释 放 这 块 内 存 ? 

在 上 面 的 代码 中 ,我们 额外 分 配 了 15 字 节 ， 在 释放 “真正 的 ”内 存 时 ， 必 须 释 放 这 块 额外 
内 存 。 

为 了 释放 整个 内 存 块 ,我 们 可 以 将 它 的 起 始 地 址 存放 在 这 块 “额外 ”内 存 中 。 我 们 会 在 紧邻 
地 址 对 齐 的 内 存 块 之 前 , 存放 这 个 地 址 。 当 然 ,， 这 意味 着 我 们 现在 需要 更 多 的 额外 内 存 ， 以 确保 
有 足够 的 空间 存放 这 个 起 始 地 址 。 

特别 是 ， 对 于 按 alignment 字 节 数 对 齐 ， 我 们 需要 额外 分 配 alignment - 1+ sizeof(void*) 
字 节 。 

下 面 是 该 做 法 的 实现 代码 。 





























1 void* aligned malloc(size t required bytes, size t alignment) { 
2 void* p1; // 原先 的 内 存 块 

3 void** p2; // 对 齐 后 的 内 存 块 

4 int offset = alignment - 1 + sizeof(void*); 

5 if ((P1 = (void*)malloc(required bytes + offset)) == NULL) { 
6 return NULL; 

7 
8 


p2 = (void**)(((size t)(p1) + offset) & ~(alignment - 1)); 


9 p2[-1] = p1; 
16 return p2; 
11 } 

12 


13 void aligned free(void *p2) { 

14 /* 为 了 一 致 性 ， 这 里 也 仿照 aligned_mal1loc 函 数 取 名 */ 
15 void* p1 = ((void**)p2)[-1]; 

16 free(p1); 

17 } 


下 面 看 看 aligned_free 是 怎么 运作 的 ， 该 函数 有 个 传人 参数 为 p2 ( 与 aligned_malloc 里 的 
p2 是 相同 的 )。 很 显然 ，p1 的 值 ( 指向 完整 内 存 块 的 开头 ) 就 存放 在 p2 的 前 面 。 

如 果 我 们 把 p2 看 作 void** (或 者 void * 的 数组 )， 就 可 以 按 索 引 - 1 取得 p1。 然 后 ， 释 放 p1 
就 可 释放 整 块 内 存 。 


13.10 用 C 编 写 一 个 my2DA11oc 国 数 ， 可 分 配 二 维 数组 。 将 malloc 子 数 的 调用 次 数 降 到 最 少 ， 
并 确保 可 通过 arr[i][j] 访 问 该 内 存 。( 第 89 页 ) 

解法 

大 家 可 能 都 知道 ,二 维 数 组 本 质 上 就 是 数组 的 数组 。 既然 可 以 用 指针 访问 数组 , 就 可 以 用 双 
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重 指针 来 创建 二 维 数组 。 

基本 思路 是 先 创建 一 个 一 维 指 针 数 组 。 然 后 ， 为 每 个 数组 索引 ， 再 新 建 一 个 一 维 数组 。 这 样 
就 能 得 到 一 个 二 维 数组 ， 可 通过 数组 索引 访问 。 

下 面 是 该 做 法 的 实现 代码 。 























1 int** my2DAlloc(int rows, int cols) { 

2 int** rowptr; 

3 int i; 

4 rowptr = (int**) malloc(rows * sizeof(int*)); 

5 for (i = 6;j i < rows; i++) { 

6 rowptr[i] = (int*) malloc(cols * sizeof(int)); 
7 } 

8 return rowptr; 


9 } 
仔细 观察 上 面 的 代码 , 注意 我 们 是 怎样 让 rowptr 根 据 索引 指向 具体 位 置 的 。 下 图 显示 了 内 存 
是 怎么 分 配 的 。 




































































释放 这 些 内 存 不 能 直接 对 rowptr 调 用 free。 我 们 要 确保 不 仅 释 放 掉 第 一 次 malloc 调 用 分 配 
的 内 存 ， 还 要 释放 后 续 每 次 malloc 调 用 分 配 的 内 存 。 





1 void my2DDealloc(int** rowptr, int rows) { 
2 for (i = 8; i < rows; i++) { 

3 free(rowptr[i]); 

4 } 

5 free(rowptr); 


6 } 
我 们 还 可 以 分 配 一 大 块 连续 的 内 存 ， 这 样 就 不 必 分 配 很 多 个 内 存 块 〈 每 一 行 一 块 ， 外 加 一 
块 内 存 ， 存 放 每 一 行 的 首 地 址 )。 举 个 例子 ， 对 于 五 行 六 列 的 二 维 数组 ， 这 种 做 法 的 效果 如 下 


图 所 示 。 
RT TT TT TT TT 


看 到 这 样 的 二 维 数组 似乎 有 点 奇怪 , 注意， 它 与 前 一 张 图 并 没什么 不 同 。 唯 一 区 别 是 现在 是 
一 大 块 连续 的 内 存 ， 因 此 ， 此 例 中 前 五 个 元 素 指向 同一 块 内 存 的 其 他 位 置 。 
下 面 是 这 种 做 法 的 具体 实现 。 


1 int** my2DAlloc(int rows, int cols) { 
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2 
3 int header = rows * sizeof(int*); 

4 int data = rows * cols * sizeof(int); 

5 int** rowptr = (int**)malloc(header + data) 
6 if (rowptr == NULL) { 

7 return NULL; 

8 


} 
9 
16 int* buf = (int*) (rowptr + rows); 
11 for (i = 6;j i < rows; i++) { 
12 rowptr[i] = buf + i * cols; 
13 } 
14 return rowptr; 
15 } 


注意 , 仔细 观察 第 11 ~ 13 行 代码 的 具体 实现 。 假 设 该 二 维 数组 有 五 行 , 每 行 六 列 , 则 array[8] 
会 指向 array[5]，array[1] 指 向 array[11] ， 依 此 类 推 。 
随后 ， 当 我 们 真正 调用 array[1][3] 时 ， 计 算 机 会 查找 array[1]， 这 是 个 指针 ， 指 向 内 存 的 
另 一 个 地 方 ， 其 实 就 是 指向 array[5] 的 指针 。 这 个 元 素 会 被 视 为 一 个 数组 ， 然 后 取出 它 的 第 3 个 
元 素 (索引 从 0 开始 )。 
用 这 种 方法 构建 数组 只 需 调 用 一 次 malloc, 另外 还 有 个 好 处 , 就 是 清除 数组 时 也 只 需 调 一 次 
free， 而 不 必 专 门 写 个 函数 释放 其 余 的 内 存 块 。 
























































9.14 Java 


14.1 ”从 继承 的 角度 来 看 ， 将 构造 函数 声明 为 私有 会 有 何 作用 ? (第 93 页 ) 


解法 

将 构造 函数 声明 为 私有 private )， 可 确保 类 以 外 的 地 方 都 不 能 直接 实例 化 这 个 类 。 在 这 种 
情况 下 , 要 创建 这 个 类 的 实例 , 唯一 的 办 法 是 提供 一 个 公共 静态 方法 , 就 像 工 厂 方法 模式 ( Factory 
Method Pattern ) 那样 。 

此 外 ， 由 于 构造 函数 是 私 有 的 ， 因 此 这 个 类 也 不 能 被 继承 。 


14.2 在 Java 中 ， 若 在 try-catch-finally 的 try 语 句 块 中 插入 return 语 句 ，finally 语 句 
块 是 否 还 会 执行 ? (第 93 页 ) 


解法 
是 的 ， 它 会 执行 。 当 退出 try 语 句 块 时 ，finally 语 句 块 将 会 执行 。 即 使 我 们 试图 从 try 语 句 
块 里 跳出 (通过 return 语 句 、continue 语 句 、break 语 句 或 任意 异常 )，finally 语 句 块 仍 将 得 以 
执行 。 

注意 ， 有 些 情况 下 finally 语 句 块 将 不 会 执行 ， 比 如 : 
口 如 果 虚 拟 机 在 try/catch 语 句 块 执行 期 间 退 出 ; 
口 如 果 执 行 try/catch 语 句 块 的 线程 被 杀 死 终止 了 。 
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14.3 ”final、finally 和 finalize 之 间 有 何 差 异 ? (第 93 页 ) 

解法 

尽管 名 字 相 像 、 发 音 类 似 ，final、finally 和 finalize 的 功能 截然 不 同 。 非 常 笼统 地 说 ， 
final 用 于 控制 变量 、 方 法 或 类 是 否 “可 更 改 "”。finally 关 键 字 用 在 try/catch 语 句 块 中 ， 以 确 
保 一 段 代码 一 定 会 被 执行 。 一 旦 垃圾 收集 需 确 定 没 有 任何 引用 指向 某 个 对 象 ,就 会 在 销毁 该 对 象 
之 前 调用 finalize() 方 法 。 

下 面 是 关于 这 几 个 关键 字 和 方法 的 更 多 细节 。 

1. final 

上 下 文 不 同 ，final 语 名 含义 有 别 。 
口 应 用 于 基本 类 型 (primitive ) 变量 时 : 该 变量 的 值 无 法 更 改 。 
口 应 用 于 引用 (reference ) 变量 时 : 该 引用 变量 不 能 指向 堆 上 的 任何 其 他 对 象 。 
口 应 用 于 方法 时 : 该 方法 不 允许 重 写 。 
口 应 用 于 类 时 : 该 类 不 能 派生 子 类 。 



































2.finally 

在 try 块 或 catch 块 之 后 ， 可 以 选择 加 一 个 finally 语 句 块 。finally 语 句 块 里 的 语句 一 定 会 
被 执行 ( 除非 Java 虚 拟 机 在 执行 try 语 句 块 期 间 退 出 )。 我 们 会 在 finally 语 句 块 里 编写 资源 回收 
和 清理 的 代码 。 


3. finalize() 
当 垃 圾 收集 器 确定 再 无 任何 引用 指向 某 个 对 象 实例 时 ， 就 会 在 销毁 该 对 象 之 前 调用 
finalize() 方 法 ,一 般 用 于 清理 资源 ， 比 如 关闭 文件 。 





14.4 “C++ 模板 和 Java 泛 型 之 间 有 何不 同 ” (第 93 页) 


解法 

许多 程序 员 都 认为 模板 (template ) 和 泛 型 ( generic ) 这 两 个 概念 是 
许 你 按照 List<string> 的 样式 编写 代码 。 不 过 ， 各 种 语言 是 怎么 实现 该 
么 做 ， 却 千差万别 。 

Java 泛 型 的 实现 植 根 于 “类 型 消除 ”这 一 概念 。 当 源 代码 被 转换 成 Java 虚 拟 机 字 节 码 时 ， 这 
种 技术 会 消除 参数 化 类 型 。 

例如 ,假设 有 以 下 Java 代 码 : 


1 Vector<String> vector = new Vector<String>(); 
2 vector.add(new String(“hello”)); 
3 String str = vector.get(0); 


编译 时 ， 上 面 的 代码 会 被 改写 为 : 


1 Vector vector = new Vector(); 
2 vector.add(new String(“hello”)); 


等 价 的 ， 因 为 两 者 都 允 
功能 的 ， 以 及 为 什么 这 
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3 String str = (String) vector.get(0); 
有 了 Java 泛 型 ， 我 们 可 以 做 的 事情 也 并 没有 真正 改变 多 少 ; 它 只 是 让 代码 变 得 漂亮 些 。 鉴 于 
此 ， Java 汉 型 有 时 也 被 称 为 “语法 糖 ”。 
这 点 跟 C++ 的 模板 截然 不 同 。 在 C++ 中 ， 模 板 本 质 上 就 是 一 套 宏 指令 集 ， 只 是 换 了 个 名 头 ， 
编译 器 会 针对 每 种 类 型 创建 一 份 模板 代码 的 副本 。 有 项 证 据 可 以 证 明 这 一 点 : MyClass<Foo> 不 
会 与 MyClass<Bar> 共 享 静态 变量 。 然 而 ， 两 个 MyClass<Foo> 实 例 则 会 共享 静态 变量 。 


看 了 下 面 的 代码 ， 应 该 会 更 清楚 一 点 
































1 /*** MyClass.h ***/ 
2 template<class T> class MyClass { 
3 public: 

4 static int val; 
5 MyClass(int v) { val = v; } 
6 ); 

6 

8 /*** MyClass.cpp ***/ 

9 template<typename T> 

16 int MyClass<T>::bar; 


12 template class MyClass<Foo>; 
13 template class MyClass<Bar>; 


15 /*** main.cpp ***/ 


16 MyClass<Foo> * fool = new MyClass<Foo>(108); 
17 MyClass<Foo> * foo2 = new MyClass<Foo>(15); 
18 MyClass<Bar> * barl1 = new MyClass<Bar>(208); 
19 MyClass<Bar> * bar2 = new MyClass<Bar>(35); 
20 

21 int fl = foo1l->val; // 等 于 15 

22 int f2 = foo2->val; // 等 于 15 

23 int bl = bar1->val;j // 等 于 35 

24 int b2 = bar2->val; // 等 于 35 


在 Java 中 ，Myclass 类 的 静态 变量 会 由 所 有 MyCclass 实 例 共 享 ， 不 论 类 型 参数 相同 与 否 。 

由 于 架构 设计 上 的 差异 ，Java 谤 型 和 C++ 模板 还 有 如 下 很 多 不 同 点 。 

口 C++ 模板 可 以 使 用 int 等 基本 数据 类 型 。Java 则 不 行 ， 必 须 转 而 使 用 Integer。 

口 在 Java 中 ， 可 以 将 模板 的 类 型 参数 限定 为 某 种 特定 类 型 。 例 如 ， 你 可 能 会 使 用 泛 型 实现 

CardDeck， 并 规定 类 型 参数 必须 扩展 自 CardGame。 

口 在 C++ 中 ， 类 型 参数 可 以 实例 化 ， 但 Java 不 支持 。 

口 在 Java 中 ， 类 型 参数 ( 即 Myclass<Foo> 中 的 Foo ) 不 外 用 于 表态 方法 和 变量 ， 因为 它们 会 
被 MyCclass<Foo> 和 MyClass<Bar> 所 共享 。 在 C++ 中 ， 这 些 类 都 是 不 同 的 ， 因 此 类 型 参数 
可 以 用 于 静态 方法 和 静态 变量 。 

口 在 Java 中 ， 不 管 类 型 参数 是 什么 ，Myclass 的 所 有 实例 都 是 同一 类 型 。 类 型 参数 会 在 运行 
时 被 抹 去 。 在 C++ 中 ， 参 数 类 型 不 同 ， 实 例 类 型 也 不 同 。 

记 住 ，Java 泛 型 和 C++ 模板 ， 虽 然 在 很 多 方面 看 起 来 都 一 样 ， 但 实则 大 不 相同 。 
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14.5 Java 中 的 对 象 反射 是 什么 ?9 它 有 什么 用 ?9 (第 93 页 ) 

解法 

对 象 反射 (Object Reflection ) 是 Java 的 一 项 特性 ， 提 供 了 获取 Java 类 和 对 象 的 反射 信息 的 方 
可 执行 如 下 操作 。 

(1) 运行 时 取得 类 的 方法 和 字段 的 相关 信息 。 

(2) 创建 某 个 类 的 新 实例 。 

(3) 通过 取得 字段 引用 直接 获取 和 设置 对 象 字 段 ， 不 管 访问 修饰 符 为 何 。 

下 面 这 段 代 码 为 对 象 反射 的 示例 。 


/* 参数 */ 
Object[] doubleArgs = new Object[] { 4.2, 3.9 }; 























/* 取得 类 */ 
Class rectangleDefinition = Class.forName(“MyProj.Rectangle”); 


/* 等 同 于 : Rectangle rectangle = new Rectangle(4.2, 3.9); */ 
Class[] doubleArgsClass = new Class[] {double.class, double.class}; 
9 Constructor doubleArgsConstructor = 

16 rectangleDefinition.getConstructor(doubleArgsClass); 

11 Rectangle rectangle = 

12 (Rectangle) doubleArgsConstructor.newInstance(doubleArgs); 


oo vam 上 wh 情 


14 /* 等 同 于 : Double area = rectangle.area(); */ 
15 Method m = rectangleDefinition.getDeclaredMethod(“area”); 
16 Double area = (Double) m.invoke(rectangle); 


这 段 代码 等 同 于 : 
1 Rectangle rectangle = new Rectangle(4.2, 3.9); 
2 Double area = rectangle.areal(); 


对 象 反射 有 什么 用 ? 

当然 ， 从 上 面 的 例子 来 看 ， 对 象 反 射 似乎 没什么 用 ， 不 过 在 特定 情况 下 反射 可 能 非常 有 用 。 

对 象 反 射 之 所 以 有 用 ， 主 要 体现 在 以 下 3 个 方面 。 

(1) 有 助 于 观察 或 操纵 应 用 程序 的 运行 时 行为 。 

(2) 有 助 于 调试 或 测试 程序 ， 因 为 我 们 可 以 直接 访问 方法 、 构 造 函 数 和 成 员 字段 。 

(3) 即使 事前 不 知道 某 个 方法 ,我 们 也 可 以 通过 名 字 调 用 该 方法 。 例 如 ， 让 用 户 传人 类 和 名、 
构造 函数 的 参数 和 方法 名 。 然 后 ,我 们 就 可 以 使 用 该 信息 来 创建 对 象 ， 并 调用 方法 。 如 果 没 有 反 
射 的 话 ， 即 使 可 以 做 到 ， 也 需要 一 系列 复杂 的 if 语 句 。 


14.6 ”实现 circularArray 类 ， 支 持 类 似 数 组 的 数据 结构 ， 这 些 数据 结构 可 以 高 效 地 进 
行 旋 转 。 该 类 应 该 使 用 泛 型 ， 并 通过 标准 的 for (0bj o : circularArray) 语 法 支持 迭代 操 
作 。( 第 93 页 ) 

解法 

此 题 实际 上 有 两 部 分 。 首 先 ， 我们 需要 实现 circularArray 类 。 其 次 , 需要 支持 迭代 。 下 面 
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将 分 别 加 以 说 明 。 


实现 CircularArray 类 
实现 CircularArray 类 的 方式 之 一 是 每 次 调用 rotate(int shiftRight) 时 都 要 移动 元 素 。 
当然 ， 这 么 做 效率 低下 。 


反之 , 我 们 可 以 只 创建 























不 必 四 处 移动 数组 元 素 ， 只 需 通 过 shiftRight 增 加 head 的 值 。 
下 面 是 该 做 法 的 实现 代码 。 


1 
2 
3 
4 
5 
6 
2 
8 


public class CircularArray<T> { 


private T[] items; 
private int head = 9; 


public CircularArray(int size) { 
items = (T[]) new Object[size]; 


} 


private int convert(int index) { 
if (index < 86) { 
index += items.length; 
} 
return (head + index) % items.1length; 


} 


public void rotate(int shiftRight) { 
head = convert(shiftRight); 


} 


public T get(int i) { 
if (i < 8 || i >= items.length) { 


throw new java.lang.IndexOutOfBoundsException(".. 


} 


return items[convert(i)]; 


;} 


public void set(int i, T item) { 
items[convert(i)] = item; 


其 中 有 几 个 地 方 很 容易 出 错 ， 比 如 : 





口 我 们 无 法 创建 泛 型 的 数组 。 相反 , 我 们 必须 将 数组 转型 或 者 将 items 类 型 
为 了 简单 起 见 ， 这 里 选用 了 前 一 种 做 法 。 
口 执行 hegValue % posVal ( 负 值 % 正 值 ) 时 








，% 操 作 符 会 


个 成 员 变量 head, 指向 概念 上 应 被 视 作 循环 数组 开头 的 元 素 。 我 们 


“"); 





Ns 


定义 为 List<T>。 


返回 负 值 。 举 个 例子 ，-8 % 3 


的 结果 为 -2。 这 跟 数 学 家 定义 的 取 模 函数 不 同 , 我 们 必须 将 负数 索引 加 上 items .length,， 

以 得 到 正确 的 正 数 值 。 
口 无 论 何 时 都 必须 确保 将 原 索引 转 成 旋转 后 的 索引 。 为 此 , 我 们 实现 了 convert 函 数 供 其 他 

函数 使 用 。 即 使 rotate 函 数 也 会 使 用 convert。 这 是 一 个 很 好 的 代码 复 用 的 范例 。 
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现在 ， 我 们 明确 了 CircularArray 的 基本 代码 ， 接 下 来 可 以 专注 于 迭代 需 的 实现 。 


实现 迭代 器 (lterator〉 接口 
此 题 的 第 二 部 分 要 求 我 们 实现 circularArray 类 之 后 ， 可 以 这 么 写 代 码 : 


1 CircularArray<String> array = ... 
2 for (String s : array) { ... } 


要 做 到 这 一 点 ， 就 必须 实现 Iterator 接 口 。 

为 实现 Iterator 接 口 ， 我 们 需要 做 到 以 下 两 点 。 

口 修改 circularArray<T> 定 义 ， 添 加 implements Iterable<T>， 同 时 还 要 在 Circular 

Array<T> 里 加 入 iterator() 方 法 。 

口 创建 实现 Iterator<T> 的 CircularArrayIterator<T>， 同 时 ， 还 要 在 CircularArray 
Iterator 里 实现 方法 hasNext() 、next() 和 remove()。 

完成 上 述 工 作 后 ，for 循 环 就 会 如 魔法 般 地 发 挥 作用 。 

为 节省 篇 幅 ， 以 下 代码 中 与 之 前 circularArray 实 现 相同 的 部 分 已 删除 。 


1 public class CircularArray<T> implements Iterable<T> { 




















2 二 

3 public Iterator<T> iterator() { 

4 return new CircularArrayIterator<T>(this); 

5 } 

6 

7 private class CircularArrayIterator<TI> implements Iterator<TI>{ 
8 /* _current 反 映 从 旋转 后 的 开头 算 起 的 偏 移 值 ， 

9 * 而 不 是 从 原始 数组 的 开头 算 起 */ 

16 private int _current = -1; 

11 private TI[] _items; 

12 

13 public CircularArrayIterator(CircularArray<TI> array)t{ 
14 _items = array.items; 

15 } 

16 

17 @Override 

18 public boolean hasNext() { 

19 return _current < items.length - 1; 

26 } 

21 

22 @Override 

23 public TI next() { 

24 _Current++; 

25 TI item = (TI) _items[convert(_current)]; 
26 return item; 

27 } 

28 

29 @Override 

36 public void remove() { 

31 throw new UnsupportedOperationException(®...”); 
32 } 

33 } 

34 } 
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注意 ， 在 上 面 的 代码 中 ， 当 for 循 环 第 一 次 迭代 时 ， 会 调用 hasNext()， 然 后 是 next()。 务 
必 确 保 你 的 实现 会 返回 正确 的 值 。 

在 面试 中 碰 到 类 似 题 目 时 , 很 有 可 能 想 不 起 来 需要 调用 哪些 方法 和 接口 。 在 这 种 情况 下 ,你 
还 是 应 该 竭尽 所 能 地 解 题 。 如 果 你 能 推导 出 可 能 需要 用 到 哪些 方法 , 光 这 样 就 能 癌 面 试 官 展现 出 
你 具备 的 能 力 。 


9.15 ”数据 库 
问题 1 ~ 3 用 到 以 下 数据 库 模 式 : 


























Apartments Buildings Tenants 
AptID int BuildingID int TenantID int 
UnitNumber varchar | ComplexID int TenantName varchar 
BuildingID | int BuildingName | varchar 

Address varchar 









































Complexes AptTenants Requests 

ComplexID int TenantID int RequestID int 

ComplexName | varchar | AptID int Status varchar 
AptID | int 
Description | varchar 














注意 ,每 套 公寓 可 能 有 多 位 承租 人 ， 而 每 位 承租 人 可 能 租 住 多 套 公寓 。 每 套 公 寓 隶 属于 一 栋 
大 楼 ， 而 每 栋 大 楼 属于 一 个 综合 体 。 


15.1 编写 SQL 查询 ， 列 出 租 住 不 止 一 套 公 寓 的 承租 人 。( 第 97 页 ) 


解法 
要 解决 此 题 ， 我 们 可 以 使 用 HAVING 和 GROUP BY 子 句 ， 然 后 将 Tenants 以 INNER JOIN 连接 
起 来 。 














SELECT TenantName 
FROM Tenants 
INNER JOIN 
(SELECT TenantID 
FROM AptTenants 
GROUP BY TenantID 
HAVING count(*) > 1) C 
ON Tenants.TenantID = C.TenantID 


在 面试 或 现实 生活 中 ， 每 当 编 写 G6ROUP BY 子 句 时 ,务必 确保 SELECT 子 句 里 的 任何 东西 ， 要 
么 是 聚集 函数 ， 要 么 就 是 包含 在 GROUP BY 子 句 里 。 


oviOmwN 
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15.2 ”编写 SQL 查 询 ， 列 出 所 有 建筑 物 ， 并 取得 状态 为 “Open” 的 申请 数量 ( Requests 
表 中 Status 为 0pen 的 条 目 )。( 第 97 页 ) 
解法 
此 题 直 接 将 Requests 和 Apartments 连 接 起 来 , 就 能 列 出 建筑 物 ID , 并 取得 Open 申 请 的 数量 。 
取得 这 份 列表 后 ， 再 将 它 与 Buildings 表 进行 连接 。 
1 SELECT BuildingName，ISNULL(Count，6) as “Count' 
FROM Buildings 


2 
3 LEFT JOIN 

4 (SELECT Apartments.BuildingID, count(*) as ‘Count?’ 
5 

6 

4 





FROM Requests INNER JOIN Apartments 
ON Requests.AptID = Apartments.AptID 
WHERE Requests.Status = ‘Open’ 
8 GROUP BY Apartments.BuildingID) ReqCounts 
9 ON ReqCounts.BuildingID = Buildings.BuildingID 


诸如 这 种 有 子 查询 的 查询 , 务必 要 经 过 全 面 测试 , 手写 时 尤 当 如 此 。 最 好 先 测试 查询 的 内 层 ， 
然后 再 测试 外 层 部 分 。 

15.3 11 号 建筑 物 正 在 进行 大 翻修 。 编 写 SQL 查 询 ， 关 闭 这 栋 建 筑 物 里 所 有 公寓 的 入 住 申 
请 。( 第 97 页 ) 

解法 

跟 SELECT 查 询 一 样 ，UPDATE 查 询 也 可 以 有 WHERE 子 句 。 要 实现 这 个 查询 ,我们 会 获取 11 号 建 
筑 物 里 所 有 公寓 的 ID ， 然 后 从 这 些 公 寓 取 得 入住 申请 列表 。 








1 _ UPDATE Requests 

2 SET Status = ‘Closed’ 

3 WHERE AptID IN 

4 (SELECT AptID 

5 FROM Apartments 

6 WHERE BuildingID = 11) 


15.4 连接 有 哪些 不 同类 型 9 请 说 明 这 些 类 型 之 间 的 差异 ， 以 及 为 何在 某 些 情形 下 ， 某 
种 连接 会 比较 好 。( 第 97 页 ) 

解法 

JOIN 用 于 合并 两 个 表 的 结果 。 要 执行 JOIN 操作 , 每 个 表 里 至 少 要 有 一 个 字段 ， 可 用 来 配对 另 
一 个 表 里 的 记录 。 连 接 的 类 型 规定 了 哪些 记录 会 进入 合并 结果 集 。 

下 面 以 两 张 表 为 例 : 一 张 表 列 出 常规 饮料 ， 另 一 张 表 是 无 卡路里 饮料 。 每 张 表 有 两 个 字段 : 
饮料 名 称 (name ) 和 产品 编号 (code )。 编 号 (code ) 字段 用 来 配对 记录 。 

常规 饮料 : 
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Name Code 
Budweiser BUDWEISER 
Coca-Cola COCACOLA 
Pepsi PEPSI 
无 卡路里 饮料 : 

Name Code 
Diet Coca-Cola COCACOLA 
Fresca FRESCA 
Diet Pepsi PEPSI 
Pepsi Light PEPSI 
Purified Water Water 











欲 将 Beverage 与 Calorie-Free Beverages 连 接 起 来 ,我们 可 以 有 多 种 选择 ， 说 明 如 下 。 

口 INNER JOIN: 结果 集 只 含有 配对 成 功 的 数据 。 在 这 个 例子 里 ， 我 们 会 得 到 三 条 记录 : 一 
条 包含 COCACOLA 编 号 ， 两 条 包含 PEPSI 编 号 。 
口 OUTER JOIN: OUTER JOIN 一 定 会 包含 TINNER JOIN 的 结果 ， 不 过 它 也 可 能 包含 一 些 在 其 他 

表 里 没有 配对 的 记录 。0OUTER JOIN 还 可 分 为 以 下 几 种 子 类 型 。 

昌 LEFT OUTER JOIN 或 简称 LEFT JOIN: 结果 会 包含 左 表 的 所 有 记录 。 如 果 右 表 中 找 不 到 
配对 成 功 的 记录 ， 则 相应 字段 的 值 为 NULL。 在 这 个 例子 里 ， 我 们 会 得 到 四 条 记录 。 除 
了 INNER JOIN 的 结果 ， 还 会 列 出 BUDNWEISER， 因 为 它 位 于 左 表 中 。 

昌 RIGHT OUTER JOIN 或 简称 RIGHT JOIN: 这 种 连接 刚好 与 LEFT JOIN 相反 。 它 会 返回 包 
括 右 表 的 所 有 记录 ; 左 表 缺 失 的 字段 为 NULL。 注 意 ， 如 果 有 两 张 表 A 和 B， 那 么 ， 可 以 
认为 语句 A LEFT JOIN B 与 B RIGHT JOIN A 等 价 。 在 上 面 的 例子 里 ， 我 们 会 得 到 五 条 
记录 。 除 了 INNER JOIN 结果 ， 还 会 有 FRESCA 和 WATER 两 条 记录 。 

昌 FULL OUTER JOIN: 这 种 连接 会 合并 LEFT 和 RIGHT JOIN 的 结果 。 不 论 另 一 个 表 里 有 无 
配对 记录 ， 这 两 个 表 的 所 有 记录 都 会 放 进 结果 集中 。 如 果 找 不 到 配对 记录 ， 则 对 应 的 
结果 字段 的 值 为 NULL。 在 这 个 例子 里 ， 我 们 会 得 到 六 条 记录 。 


15.5 ”什么 是 反 规 范 化 ? 请 说 明 优 缺点 。( 第 97 页 ) 


解法 

反 规 范 化 ( denormalization ) 是 一 种 数据 库 优 化 技术 ， 在 一 个 或 多 个 表 中 加 入 宛 余 数据 。 在 
使 用 关系 型 数据 库 中 ， 反 规范 化 可 帮助 我 们 避免 开销 很 大 的 表 连 接 操作 。 

相 比 之 下 , 在 传统 的 规范 化 数据 库 中 , 我 们 会 将 数据 存放 在 不 同 的 逻辑 表 里 ， 试 图 将 元 余数 
据 减 到 最 少 ， 力 争 做 到 在 数据 库 中 每 块 数据 只 有 一 份 副 本 。 

例如 ， 在 规范 化 数据 库 中 ， 我 们 可 能 会 有 Courses 表 和 Teachers 表 。 在 Courses 里 ， 每 个 条 
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目 都 会 储存 课程 ( Course ) 的 teacherID, 但 不 存储 teacherName。 如 和 欲 获取 所 有 课程 (Courses ) 
对 应 的 教师 (Teacher ) 姓名 ， 只 需 对 这 两 个 表 进 行 连接 。 

就 某 些 方面 来 看 ， 这 人 么 做 很 不 错 。 如 有 教师 更 改名 字 ， 我 们 只 需 更 新 一 个 地 方 的 名 字 。 

不 过 ， 这 人 么 做 的 缺点 在 于 ， 如 果 表 很 大 ， 就 需要 花费 过 长 时 间 对 这 些 表 执 行 连接 操作 。 

而 反 规范 化 则 可 以 达成 一 定 的 平衡 。 在 反 规范 化 时 ， 我们 确定 自己 可 以 接受 一 定 的 匈 余 , 并 
在 更 新 数据 库 时 要 多 做 些 工作 ， 从 而 减少 连接 操作 ， 保 证 较 高 的 效率 。 






































反 规 范 化 的 缺点 反 规 范 化 的 优点 
更 新 和 插入 操作 开销 更 大 ee 
检索 数据 更 快 





代码 更 难 写 


反 规范 化 会 使 更 新 和 插入 








需要 查找 的 表 较 少 ， 因 此 检索 查 
询 比 较 简 单 (因而 也 不 容易 出 错 ) 
































据 可 能 不 一 致 。 哪 一 块 数 据 





昌 存 在 元 余 
要 更 大 的 存储 空间 














数 
才 是 “正确 ”的 呢 ? 
数 
需 


» 
































在 注重 可 扩展 性 的 系统 中 ， 比 如 大 型 科技 公司 ,几乎 一 定 会 兼用 规范 化 和 反 规 范 化 数据 库 的 


各 种 要 素 。 





15.6 ”有 个 数据 库 ， 里 面 有 公司 (companies )、 人 (people ) 和 专业 人 员 (professionals ， 
为 公司 工作 )， 请 绘制 实体 关系 图 。( 第 97 页 ) 


解法 


在 公司 ( Companies ) 上 班 的 人 (People ) 称 作 专 业 人 员 (Professional )。 因 此 ，People 
和 Professional 之 间 是 ISA (“is a”) 关系 (或 者 说 Professional 派 生 自 People )。 
除了 从 People 派 生 的 属性 ，Professional 还 有 一 些 附加 信息 ,包括 学 历 ( degree ) 和 工作 经 





验 (experience ) 等 。 





每 位 Professional 同 一 时 间 只 能 为 一 家 Company 工 作 ， 而 Companies 则 可 以 同时 雇佣 多 位 
Professional， 因 此 Professional 和 Companies 之 间 是 多 对 一 的 关系 。“Works For” 关 系 可 以 


存放 员工 的 入 职 时 间 和 薪资 等 
定义 。 


一 个 People 可 能 拥有 多 个 电话 号 码 ， 所 以 phone 是 个 多 值 属 性 。 





属性 。 这 些 























属性 只 有 在 将 Professional 与 Company 相 关联 时 才 会 
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People 











ISA 





Professional 











< Degree > 





Joining 


Phone 





Companies | 

















15.7 给 定 一 个 储存 有 学 生成 绩 的 简单 数据 库 。 设 计 这 个 数据 库 的 大 概 样子 ， 并 编写 SQL 查 
询 ， 返 回 优 等 生 名 单 (排名 前 10% )， 以 平均 分 排序 。( 第 97 页 ) 


解法 


在 一 个 简单 的 数据 库 中 ， 最 起 码 会 有 三 个 对 象 : Students ( 学生 )、Courses (课程 ) 和 
CourseEnrollment ( 选修 课程 )。Sstudents 至 少 会 有 学 生 姓 名 、 学 号 (ID )， 还 可 能 包含 其 他 个 
人 信息 。Courses 会 包含 课程 名 和 代号 ,或许 还 有 课程 说 明 、 教 授 和 其 他 信息 。CourseEnrollment 
会 将 students 和 courses 配 对 起 来 ， 还 会 含有 Grade" 字 段 。 








Q@ 原文 为 CourseGrade。 
















































































Students 
StudentID int 
StudentName varchar(168) 
Address varchar(586) 
Courses 
CourseID int 
CourseName varchar(166) 
ProfessorID int 
CourseEnrollment 
CourseID Int 
StudentID int 
Grade float 
Term int 

译 者 注 
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要 是 加 上 教授 的 资料 、 学 分 费用 信息 和 其 他 数据 ， 这 个 数据 库 就 会 变 得 相当 复杂 。 





/* 错误 代码 */ 

SELECT TOP 16 PERCENT AVG(CourseEnrollment.Grade) AS GPA, 
CourseEnrollment.StudentID 

FROM CourseEnrollment 

GROUP BY CourseEnrollment.StudentID 

ORDER BY AVG(CourseEnrollment.Grade) 


QQ 上 WwW 





使 用 微软 SQL Server 里 的 TOP . .PERCENT 函数 ， 我 们 可 以 先 党 试 如 下 〈 错误 的 ) 查询 : 


以 上 代码 的 问题 在 于 ， 它 只 会 如 实 返 回 按 GPA 排 序 后 的 前 10% 行 记录 。 设 想 这 样 一 个 场景 : 





有 100 名 学 生 ， 排 名 前 15 的 学 生 的 GPA 都 是 4.0。 上 面 的 函数 只 会 返回 其 中 10 名 学 生 ， 与 我 们 的 要 
求 不 符 。 在 得 分 相同 的 情况 下 ， 我们 希望 计 入 得 分 前 10% 的 学 生 ， 即 使 优等 生 名 单 的 人 数 超过 班 


级 总 人 数 的 10%. 





为 纠正 这 个 问题 ， 我 们 可 以 建立 类 似 的 查询 ， 不 过 首先 要 取得 筛选 优等 生 的 GPA 基 准 。 


1 DECLARE @GPACutOff float; 
2 SET @GPACUtOff = (SELECT min(GPA) as “GPAMin” 
3 FROM ( 


SELECT TOP 16 PERCENT AVG(CourseEnrollment.Grade) AS GPA 


4 
5 FROM CourseEnrollment 

6 GROUP BY CourseEnrollment .StudentID 
村 ORDER BY GPA desc) 

8 Grades); 


接着 ， 定 义 好 @GPACutoff 后 ， 要 筛选 最 低 拥 有 该 GPA 的 学 生 ， 


1 SELECT StudentName, GPA 
FROM ( 
SELECT AVG(CourseEnrollment.Grade) AS GPA, 
CourseEnrollment.StudentID 
FROM CourseEnrollment 
GROUP BY CourseEnrollment.StudentID 


Nm 上 和 wNP 


就 相当 容易 了 。 


HAVING AVG(CourseEnrollment .Grade) >= @GPACUtOff) Honors 


8 INNER JOIN Students ON Honors.StudentID = Student .StudentID 
对 于 你 所 做 出 的 隐 舍 假设 条 件 ， 必须 非 常 小 心 。 仔 细 查 看 上 面 的 数据 库 描述 ,你 会 发 现 哪 些 
可 能 是 不 正确 的 假设 ?其 中 之 一 是 每 门 课程 只 能 由 一 位 教授 来 教 。 而 在 某 些 学 校 , 一 门 课程 可 能 








会 由 多 位 教授 来 教 。 


不 过 ,你 还 是 需要 做 出 一 些 假设 , 要 不 然 会 把 自己 搞 狗 。 相 比 你 做 了 哪些 假设 , 更 重要 的 是 
认识 到 自己 做 出 了 假设 。 不 论 是 在 实际 操作 还 是 面试 中 ,就 算 假设 条 件 不 正确 ， 只 要 可 以 识别 出 


























来 ， 就 能 予以 妥善 处 理 。 





此 外 , 请 记 住 ， 弹 性 和 复杂 度 之 间 需 要 权衡 取舍 。 若 建立 的 系统 支持 一 门 课程 可 由 多 位 教授 





来 教 , 的确 会 增加 数据 库 的 弹性 , 但 又 徒 增 其 复杂 度 。 倘 若 要 让 数据 
最 终 数据 库 只 会 变 得 复杂 不 堪 。 

尽量 让 你 的 设计 保持 合理 的 弹性 , 并 陈 明 任何 其 他 的 假设 或 限 4 
设计 ， 对 于 面向 对 象 设计 和 常规 的 编程 同样 适用 。 
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库 灵 活 应 对 各 种 可 能 的 情况 ， 





由 条 件 。 这 不 仅 适用 于 数据 库 











9.16 ”线程 与 锁 


16.1 ”线程 和 进程 有 何 区 别 ? (第 103 页) 


解法 

进程 和 线程 彼此 有 关联 ， 但 两 者 有 着 根本 上 的 不 同 。 

进程 可 以 看 作 是 程序 执行 时 的 实例 ， 是 一 个 分 配 了 系统 资源 ( 比如 CPU 时 间 和 内 存 ) 的 独立 
实体 。 每 个 进程 都 在 各 自 独立 的 地 址 空间 里 执行 , 一 个 进程 无 法 访问 另 一 个 进程 的 变量 和 数据 结 
构 。 如 果 一 个 进程 想 要 访问 其 他 进程 的 资源 ， 就 必须 使 用 进程 间 通 信 机 制 ,， 包括 管道 、 文 件 、 套 
接 字 (socket ) 及 其 他 形式 。 

线程 存在 于 进程 中 ， 共 享 进程 的 资源 (包括 它 的 堆 空 间 )。 同 一 进程 里 的 多 个 线程 将 共享 同 
一 个 堆 空 间 。 这 中 进程 大 不 相同 ， 一 个 进程 不 能 直接 访问 另 一 个 进程 的 内 存 。 不 过 ,每 个 线程 仍 
然 会 有 自己 的 寄存 器 和 栈 ， 而 其 他 线程 可 以 读 写 堆 内 存 。 

线程 是 进程 的 某 条 执行 路 径 。 当 某 个 线程 修改 进程 资源 时 ,其 他 兄弟 线程 就 会 立即 看 到 由 此 
产生 的 变化 。 


16.2 ”如 何 测量 上 下 文 切换 时 间 ? (第 103 页 ) 


解法 

此 题 比较 琼 手 ， 我 们 不 妨 先 从 一 种 可 能 的 解法 入 手 。 

上 下 文 切 换 ( context switch ) 是 两 个 进程 之 间 切 换 (也 即 ， 将 等 待 中 的 进程 转 为 执行 状态 ， 
而 将 正在 执行 的 进程 转 为 等 待 或 终止 状态 ) 所 耗费 的 时 间 。 这 样 的 动作 会 发 生 在 多 任务 处 理 系统 
中 ， 操 作 系 统 必须 将 等 待 中 进程 的 状态 信息 载 入 内存， 并 保存 执行 中 进程 的 状态 信息 。 

为 了 解决 此 题 , 我 们 需要 记录 两 个 交换 进程 执行 最 后 一 条 和 第 一 条 指令 的 时 间 戳 ,而 上 下 文 
切换 时 间 就 是 这 两 个 进程 的 时 间 戳 差 值 。 

举 个 简单 的 例子 : 假设 只 有 两 个 进程 P, 和 P,。 

Pl 正在 执行 ，P; 则 在 等 待 执 行 。 在 某 一 时 间 点 ， 操 作 系 统 必须 交换 P| 和 P,， 假 设 正好 发 生 在 
Pl 执 行 第 N 条 指令 之 际 。 若 tx 表示 进程 x 执行 第 k 条 指令 的 时 间 戳 ， 单 位 为 微 秒 ， 则 上 下 文 切 换 需 
要 bi 一 fin 微 秒 。 

此 题 坏 手 的 地 方 在 于 : 如 何 知道 两 个 进程 何 时 会 进行 交换 呢 ? 当然 , 我 们 无 法 记录 进程 每 条 
指令 的 时 间 惟 。 

还 有 一 个 问题 , 进程 交换 是 由 操作 系统 的 调度 算法 负责 的 , 另外 还 可 能 有 很 多 内 核 态 线程 也 
会 进行 上 下 文 切换 。 其 他 进程 也 可 能 会 竞争 CPU， 或 者 内 核 还 要 处 理 中 断 ， 用 户 控制 不 了 这 些 不 
相干 的 上 下 文 切换 。 举 例 来 说 , 若 内 核 在 fu 时刻 决定 处 理 某 个 中 断 , 那么 ， 上 下 文 切换 时 间 就 会 
比 预 估 的 更 长 。 

为 克服 这 些 障碍 ,我 们 必须 先 构造 一 个 环境 : 在 P| 执行 之 后 ,任务 调度 器 会 立即 选中 并 执行 
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P2。 具 体 做 法 是 在 P; 和 P; 之 间 构 造 一 条 数据 通道 ， 如 管道 ， 证 这 两 个 进程 玩 一 场 数据 令 牌 的 桌球 
游戏 。 

换言之 ,我 们 让 P, 作 为 初始 发 送 方 ，P 作 为 接收 方 。 一 开始 ， Ps 阻塞 睡眠 ) 等 待 获取 数 据 
令 牌 。P1 执 行 时 会 将 令 牌 通过 数据 通道 递送 给 P,， 并 立即 尝试 读 取 响 应 令 牌 。 然 而 ， 由 于 Ps 还 没 
有 机 会 执行 ， 因 此 Pi 收 不 到 这 个 响应 令 牌 ， 继 而 被 阻塞 并 释放 CPU。 

随 之 而 来 的 就 是 上 下 文 切 换 , 任务 调度 器 必须 选择 另 一 个 进程 执行 。P: 正 好 处 于 随时 可 执行 
的 状态 ， 因 此 也 就 顺理成章 地 成 为 任务 调度 器 可 选择 执行 的 理想 候选 者 。 当 P: 执 行 时 ，P; 和 P2 的 
角色 互 换 了 。 现 在 ，P:? 成 为 发 送 方 ， 而 Pi 成 为 被 阻塞 的 接收 方 。 当 P: 将 令 牌 返回 给 Pi 时 ， 游 戏 即 


.Ew 
告 结 

















简 而 言 之 ， 这 个 游戏 一 个 来 回 由 以 下 步 又 组 成 。 
(1) P; 阻 寒 ， 等待 Pl 发 送 的 数据 。 
(2) Pi 标记 开始 时 间 。 
(3) Pi 向 P; 发 送 令 牌 。 
(4) P1 试 着 读 取 P, 发 送 的 响应 令 牌 ， 引 发 上 下 文 切换 。 
(5) P; 被 调度 执行 ， 接 收 P; 发 送 的 令 牌 。 
(6) P? 向 Pi 发 送 响应 令 牌 。 
(7) P; 试 着 读 取 P 发 送 的 响应 令 牌 ， 引 发 上 下 文 切换 。 
(8) Pi 被 调度 执行 ， 接 收 P; 发 送 的 令 牌 。 
(9) Pi 标记 结束 时 间 。 
这 里 的 关键 在 于 数据 令 牌 的 发 送 会 引发 上 下 文 切换 。 令 Ta 和 了 T 分 别 为 发 送 和 接收 数据 令 牌 的 
时 间 ， 并 令 T. 为 上 下 文 切换 耗费 的 时 间 。 在 第 (2) 步 , Pi 会 记录 令 牌 发 送 的 时 间 难 ,而 在 第 (9) 步 则 
记录 了 令 牌 响应 的 时 间 戳 。 这 两 个 事件 之 间 用 掉 的 时 间 T 表 示 如 下 : 
T=2*(Ty+ T+T,) 

这 个 算式 由 以 下 事件 组 成 : Pi 发 送 一 个 令 牌 3)，CPU 上 下 文 切换 (4)，P, 接 收 这 个 令 牌 ($)。 
随后 ，P; 发 送 响应 令 牌 (0)，CPU 上 下 文 切 换 (7)， 最 后 P; 收 到 这 个 啊 应 令 牌 (8)。 

接着 ， 由 Pi 很 容易 就 能 计算 T， 即 事件 3 和 事件 8 之 间 经 过 的 时 间 。 总 之 ， 若 想 求 出 T。， 我 们 
必须 先 确定 Ts+ ,的 值 。 

该 怎么 做 呢 ?” 我 们 可 以 测量 P1 发 送 和 接收 令 牌 所 耗费 的 时 间 多 少 。 不 过 这 不 会 引发 上 下 文 切 
换 ， 因 为 发 送 这 个 令 牌 时 Pl 正在 CPU 中 执行 ， 而 且 接 收 时 也 不 会 处 于 阻塞 状态 。 

将 上 述 游戏 重复 玩 多 个 来 回 , 以 剔除 步骤 (2) 和 (9) 之 间 可 能 因 意 料 之 外 的 内 核 中 断 和 其 他 内 
核 线程 对 CPU 的 竞争 而 引 和 人 的 时 间 变 动 。 我 们 将 选择 测 得 的 最 短 上 下 文 切 换 时 间作 为 最 终 
答案 。 

话说 回来 ， 最 后 我 们 只 能 说 ， 这 只 是 近似 值 ， 而 且 取 决 于 底层 系统 。 比 如 ， 我们 做 了 这 样 的 
假设 : 一 旦 数据 令 牌 可 用 , P; 就 会 被 选中 并 执行 。 而 实际 上 , 这 要 取决 于 任务 调度 器 的 具体 实现 ， 
我 们 无 法 做 出 任何 保证 。 
没关系 , 就 算 这 样 也 不 要 其 。 在 面试 中 , 能 够 意识 到 你 的 解法 或 许 不 够 完美 , 这 一 点 很 重要 。 
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16.3 在 著名 的 哲学 家 就 餐 问 题 中 ， 一 群 哲 学 家 围 坐 在 圆桌 周围 ， 每 两 位 哲学 家 之 间 有 一 
根 生子 。 每 位 哲学 家 需要 两 根 筷 子 才能 用 餐 ， 并 且 一 定 会 先 拿 起 左手 边 的 筷子 ， 然 后 才 会 去 拿 
右手 边 的 筷子 。 如 果 所 有 哲学 家 在 同一 时 间 拿 起 左手 边 的 筷子 ， 就 有 可 能 造成 死 锁 。 请 使 用 线 
程 和 锁 ， 编 写 代码 模拟 哲学 家 就 餐 问 题 ， 避 免 出 现 死 锁 。( 第 103 页 ) 


解法 
首先 ， 先 不 管 死 锁 , 让 我 们 写 些 代码 简单 模拟 哲学 家 就 餐 问题 。 具体 实现 时 ， 从 Thread 派 生 
Philosopher，Chopstick 被 拿 起 来 时 会 调用 lock.1lock()， 放 下 时 则 调用 lock.unlock()。 











1 public class Chopstick { 

2 private Lock lock; 

3 

4 public Chopstick() { 

5 lock = new ReentrantLock(); 
6 } 

7 

8 public void pickUp() { 
9 void lock.lock(); 

16 } 

11 

12 public void putDown() { 
13 lock.unlock(); 

14 } 

15 } 

16 


17 public class Philosopher extends Thread { 
18 private int bites = 108; 
19 private Chopstick left; 
26 private Chopstick right; 


21 

22 public Philosopher(Chopstick left, Chopstick right) { 
23 this.left = left; 

24 this.right = right; 
25 } 

26 

27 public void eat() { 

28 pickUp(); 

29 chew( ); 

36 putDown( ) ; 

31 } 

32 

33 public void pickUp() { 
34 left.pickUp(); 

35 right.pickUp(); 

36 } 

37 

38 public void chew() { } 
39 

40 public void putDown() { 
41 left.putDown(); 

42 right.putDown(); 
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43 } 


45 public void run() { 

46 for (int i = 6;j i < bites; i++) { 
47 eat(); 

48 } 





如 果 所 有 哲学 家 都 拿 起 左手 边 的 一 根 筷子 , 并 都 等 着 拿 右手 边 的 男 一 根 筷子 , 运行 上 面 的 代 
码 就 可 能 造成 死 锁 。 
为 了 防止 发 生死 锁 , 我 们 的 实现 可 以 采用 如 下 策略 : 如 有 哲学 家 拿 不 到 右手 边 的 筷子 ， 就 让 
他 放下 已 拿 到 的 左手 边 的 筷子 。 
public class Chopstick { 
/* 同 前 */ 


public boolean pickUp() { 
return lock.tryLock(); 
} 
} 


9 public class Philosopher extends Thread { 
16 /* 同 前 */ 


11 

12 public void eat() { 

13 if (pickUp()) { 

14 chew(); 

15 putDown(); 

16 } 

17 } 

18 

19 public boolean pickUp() { 
26 /* 试 着 拿 起 和 化 子 */ 

21 if (!left.pickUp()) { 
22 return false; 

23 } 

24 if (!right.pickUp()) { 
25 left.putDown(); 

26 return false; 

27 } 

28 return true; 

29 } 

30 } 








在 上 面 的 代码 里 ,要 确保 拿 不 到 右手 边 的 簧 子 时 就 要 放下 左手 边 的 筷子 ; 如 果 手 上 根本 没有 
筑 子 ， 就 不 该 调用 putDown() 。 


16.4 ”设计 一 个 类 ， 只 有 在 不 可 能 发 生死 锁 的 情况 下 ， 才 会 提供 锁 。〈 第 103 页 ) 





解法 
防止 死 锁 有 几 种 常见 的 方法 , 其 中 常用 的 做 法 之 一 是 , 要 求 进程 事先 声明 它 需 要 哪些 锁 。 然 9 
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n 四 
中 


了 Ed 2} 


这 可 能 会 造成 死 锁 ， 因 为 ， 存 在 以 下 场景 : 


后 ， 就 可 以 加 以 验证 ， 若 提供 锁 是 否 会 造成 死 锁 ， 会 的 话 就 不 提供 。 
间 记 这 些 限制 条 件 ， 下 面 来 探讨 如 何 检 测 死 锁 。 假 设 多 个 锁 被 请 求 的 顺序 如 下 : 





我 们 可 以 将 上 面 的 场景 看 作 一 个 图 ， 其 中 2 连接 到 3 、3 连 接 到 5， 而 5 连接 到 2。 死 锁 会 由 环 表 
示 。 如 果 某 个 进程 声明 它 会 在 锁 住 w 后 立即 请 求 锁 v， 则 图 里 就 会 存在 一 条 边 (w，v) 。 以 先前 的 
例子 来 说 ， 在 图 里 会 存在 下 面 这 些 边 : (1, 2)、(2, 3)、(3, 4)、(1, 3)、(3, 5)、(7, 5)、 











(5，9)、(9，2)。 人 至 于 这 些 边 的 “所 有 者 ”是 谁 并 不 重要 。 





这 个 类 需要 一 个 declare 方 法 ,线程 和 进程 会 以 该 方法 声明 它们 请 求 资源 的 顺序 。 这 个 


declare 方 法 将 迭代 访问 声明 顺序 ， 将 邻近 的 每 对 元 素 (v,，w) 加 





到 图 里 。 然 后 ， 它 会 检查 是 否 存 


在 环 。 如 果 存 在 环 ， 它 就 会 原 路 返回 ， 从 图 中 移 除 这 些 边 ， 然 后 退出 。 











现在 只 剩 下 一 部 分 有 竺 探讨: 如 何 检测 有 无 环 ? 我 们 可 以 通过 对 每 个 连接 起 来 的 部 分 (也 就 








接 的 部 分 ,但 那样 就 会 更 复杂 了 。 就 此 题 而 言 ， 还 没 必 要 复杂 至 


是 图 中 每 个 连接 在 一 起 的 部 分 ) 执行 深度 优先 搜索 来 检测 有 没有 环 。 也 有 算法 能 选择 图 中 所 有 连 





I 这 个 程度 。 


我 们 可 以 确定 ， 如果 出 现 了 环 ， 就 表明 是 某 一 条 新 加 入 的 边 造 成 的 。 这 样 一 来 ,只 要 深度 优 





先 搜索 会 探测 所 有 这 些 边 ， 就 等 同 于 做 过 完整 的 搜索 。 
这 种 特殊 的 环 的 检测 算法 ， 其 伪 码 如 下 所 示 : 


1 boolean checkForCycle(locks[] locks) { 

2 touchedNodes = hash table(lock -> boolean) 

3 initialize touchedNodes to false for each lock in lo 
4 for each (lock x in process.locks) { 

5 if (touchedNodes[x] == false) { 

6 if (hasCycle(x, touchedNodes)) { 

4 return true; 

8 


} 
9 ’ 
16 
11 return false; 
Ep 
13 


14 boolean hasCycle(node x, touchedNodes) { 
15 touchedNodes[r] = true; 
16 if (x.state == VISITING) { 


7 return true; 

18 } else if (x.state == FRESH) { 
19 ... (see full code below) 

26 } 

2 
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注意 ， 在 上 面 的 代码 中 ， 可 能 需要 执行 几 次 深度 优先 搜索 ， 但 touchedNodes 只 会 初始 化 一 
次 。 我 们 会 不 断 兴 代 ， 直 至 touchedNodes 中 所 有 值 都 变 为 false。 

下 面 的 代码 提供 了 更 多 细节 。 为 了 简单 起 见 , 我 们 假设 所 有 锁 和 进程 (所 有 者 ) 都 是 按 顺序 
排列 的 。 





1 public class LockFactory { 

2 private static LockFactory instance; 

3 

4 private int numberOfLocks = 5; /* 缺 尖 */ 

5 private LockNode[] locks; 

6 

区 /* 从 一 个 进程 或 所 有 者 映射 到 该 

8 * 所 有 者 宣称 它 会 要 求 锁 的 顺序 */ 

9 private Hashtable<Integer, LinkedList<LockNode>> lockOrder; 
16 

11 private LockFactory(int count) { ... } 

12 public static LockFactory getInstance() { return instance; } 
13 

14 public static synchronized LockFactory initialize(int count) { 
15 if (instance == null) instance = new LockFactory(count); 
16 return instance; 

17 } 

18 

19 public boolean hasCycle( 

20 Hashtable<Integer, Boolean> touchedNodes, 

21 int[] resourcesInOrder) { 

22 /* 检查 有 无 环 */ 

23 for (int resource : resourcesInOrder) { 

24 if (touchedNodes.get(resource) == false) { 

25 LockNode n = locks[resourcel]; 

26 if (n.hasCycle(touchedNodes)) { 

27 return true; 

28 } 

29 } 

36 } 

31 return false; 

32 } 

33 


34 /* 为 了 避免 死 锁 ， 强 制 每 个 进程 都 要 事先 宣告 
35 * 它们 要 求 锁 的 顺序 。 验 证 这 个 顺序 不 会 形成 
36 * 死 锁 (在 有 向 图 里 出 现 

37 * 环 ) */ 





38 public boolean declare(int ownerId, int[] resourcesInOrder) { 
39 Hashtable<Integer, Boolean> touchedNodes = 

46 new Hashtable<Integer, Boolean>(); 

41 

42 /* 将 结 点 加 入 图 中 */ 

43 int index = 1; 

44 touchedNodes .put(resourcesInOrder[06], false); 

45 for (index = 1; index < resourcesInOrder.length; index++) { 
46 LockNode prev = locks[resourcesInOrder[index - 1]]; 

47 LockNode curr = locks[resourcesInOrder[index]]; 
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48 prev.joinTo(curr); 

49 touchedNodes.put(resourcesInOrder[index], false); 
56 } 

5 

52 /* 如 果 出 现 了 环 ， 销 毁 这 份 资 源 列表 ， 并 

53 * 返回 false */ 

54 if (hasCycle(touchedNodes, resourcesInOrder)) { 

55 for (int j = 1; j < resourcesInOrder.length; j++) { 
56 LockNode p = locks[resourcesInorder[j - 1]]; 
57 LockNode c = locks[resourcesInOrder[j]]; 

58 p.remove(c); 

59 } 

66 Peturn false; 

61 } 

62 

63 /* 为 检测 到 环 ， 保 存 宣告 的 顺序 ， 以 便 

64 * 验证 该 进程 确实 按照 它 宣 称 的 顺序 要 求 

65 * 锁 */ 

66 LinkedList<LockNode> list = new LinkedList<LockNode>(); 
67 for (int i = 6@; i < resourcesInOrder.length; i++) { 
68 LockNode resource = locks[resourcesInOrder[i]]; 
69 list.add(resource); 

76 } 

71 lockOrder.put(ownerId, list); 

72 

73 return true; 

74 } 

了 5 


76 /* 取得 锁 ， 首 先 验证 该 进程 确实 按照 它 宣 告 的 顺序 
了 7 * 要 求 锁 */ 
78 public Lock getLock(int ownerId, int resourceID) { 


79 LinkedList<LockNode> list = lockOrder.get(ownerId); 
86 if (list == null) return null; 

81 

82 LockNode head = list.getFirst(); 
83 if (head.getId() == resourceID) { 
84 list.removeFirst(); 

85 return head.getLock(); 

86 } 

87 return null; 

88 } 

89 } 

90 


91 public class LockNode { 
92 public enum VisitState { FRESH, VISITING, VISITED }; 


94 private ArrayList<LockNode> children; 
95 private int lockId; 
96 private Lock lock; 


97 private int maxLocks; 

98 

99 public LockNode(int id, int max) { ... } 
166 


161  /* 连接 “this” 结 点 与 “node” 结 点 ， 检 查 以 确保 这 么 做 不 会 
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162 * 形成 环 */ 
163 public void joinTo(LockNode node) { children.add(node); } 


164 public void remove(LockNode node) { children.remove(node); } 
165 


166  /* 以 深度 优先 搜索 检查 是 否 存 在 环 */ 
167 public boolean hasCycle( 


168 Hashtable<Integer, Boolean> touchedNodes) { 
169 VisitState[] visited = new VisitState[maxLocks]; 
116 for (int i = 6;j i < maxLocks; i++) { 

111 visited[i] = VisitState.FRESH; 

112 } 

113 return hasCycle(visited, touchedNodes); 

114 } 

115 

116 private boolean hasCycle(VisitState[] visited， 
117 Hashtable<Integer, Boolean> touchedNodes) { 
118 if (touchedNodes.containsKey(lockId)) { 

119 touchedNodes .put(lockId, true); 

126 } 

121 

122 if (visited[lockId] == VisitState.VISITING) { 
123 /* 还 在 访问 时 却 回 到 了 这 个 结 点 ， 

124 * 表明 有 环 */ 

125 return true; 

126 } else if (visited[lockId] == Visitstate.FRESH) { 
127 visited[lockId] = VisitState.VISITING; 

128 for (LockNode n : children) { 

129 if (n.hasCycle(visited, touchedNodes)) { 
136 return true; 

131 } 

132 } 

133 visited[lockId] = VisitState.VISITED; 

134 } 

135 return false; 

136 } 

137 

138 public Lock getLock() { 

139 if (lock == null) lock = new ReentrantLock(); 
146 return lock; 

141 } 

142 

143 public int getId() { return lockId; } 

144 } 


如 同 以 往 , 当 你 看 到 这 上段 既 复杂 又 宛 长 的 代码 时 , 就 会 明白 面试 官 一 般 不 会 要 求 你 写 出 全 部 
代码 。 更 有 可 能 的 情况 是 ， 面 试 官 会 要 求 你 勾勒 出 伪 码 ， 并 实现 其 中 一 个 方法 。 


16.5 ”给 定 以 下 代码 : 


public class Foo { 








public Foo() { ... } 

public void first() { ... } 
public void second() { ... } 
public void third() { ... } 
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同一 个 Foo 实 例会 被 传 入 3 个 不 同 的 线程 。threadA 会 调用 first，threadB 会 调用 second， 
threadC 会 调用 third。 设 计 一 种 机 制 ， 确 保 first 会 在 second 之 前 调用 ，second 会 在 third 之 前 
调用 。( 第 103 页 ) 

解法 

一 般 的 逻辑 是 检查 在 执行 second() 之 前 first() 是 否 已 完成 ， 在 调用 third() 之 前 second() 
是 否 已 完成 。 我 们 必须 小 心 处 理 线程 安全 ， 因 此 ， 简 单 的 布尔 标志 达 不 到 要 求 。 

那么 ， 以 如 下 代码 来 使 用 锁 ， 怎 么 样 ? 



































1 public class FooBad { 

2 public int pauseTime = 1666; 

3 public ReentrantLock lock1，1lock2，1lock3; 
4 

5 public FooBad() { 

6 try { 

7 lock1 = new ReentrantLock(); 

8 lock2 = new ReentrantLock(); 

9 lock3 = new ReentrantLock(); 

16 

二 lock1.1lock(); 

12 lock2.1ock(); 

13 lock3.1lock(); 

14 } -cateh (Caw) € in} 

15 } 

16 

17 public void first() { 

18 try { 

19 i 

26 lock1.unlock(); // 标记 first() 已 完成 
21 } cateh (ses) € nar } 

22 } 

3 

24 public void second() { 

25 try { 

26 lock1.lock(); // 等 待 ， 直 到 first() 完 成 
27 lock1.unlock(); 

28 

29 

36 lock2.unlock(); // 标记 second() 已 完成 
31 Fatehi (ea) Ci 

32 } 

33 

34 public void third() { 

35 try { 

36 lock2.lock(); // 等 待 ， 直 到 second() 完 成 
37 lock2.unlock(); 

38 a 

39 } catch (...){...} 

40 } 

41 } 
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这 段 代 码 实际 上 并 不 满足 题目 要 求 , 关键 在 于 锁 的 所 有 权 这 个 概念 。 真正 请 求 锁 的 是 一 个 线 
程 (在 FooBad 构 造 函 数 中 )， 而 释放 锁 的 却 是 另 一 个 线程 。 这 么 做 是 不 允许 的 ， 这 段 代 码 会 抛 出 
异常 。 在 Java 中 ， 锁 的 所 有 者 和 拿 到 锁 的 线程 必须 是 同一 个 。 

换 种 做 法 ， 我 们 可 以 用 信和 号 量 重 现 这 一 行为 ， 整 个 逻辑 完全 相同 。 


1 public class Foo { 

2 public Semaphore sem1，sem2，sem3; 
3 

4 public Foo() { 

5 try { 

6 sem1 = new Semaphore(1); 
7 sem2 = new Semaphore(1); 
8 sem3 = new Semaphore(1); 
9 

16 sem1.acquire(); 

11 sem2.acquire(); 

12 sem3.acquire(); 

13 }- eateh (C0) € uv} 

14 } 

15 

16 public void first() { 

17 try { 

18 yr 

19 seml.release(); 

26 jateh ee 和 小 

21 } 

22 

23 public void second() { 

24 try { 

25 sem1.acquire(); 

26 seml.release(); 

27 让 

28 sem2.release(); 

29 } catch (...){...} 

36 } 

31 

32 public void third() { 

33 try { 

34 sem2.acquire(); 

3 sem2.release(); 

36 

37 } catch (...){...} 

38 

39 } 


16.6 ”给 定 一 个 类 ， 内 合同 步 方法 和 A 和 普通 方法 B。 在 同一 个 程序 实例 中 ， 有 两 个 线程 ， 能 
否 同 时 执行 A? 两 者 能 否 同 时 执行 A 和 B? (第 104 页 ) 

解法 

在 方法 前 加 上 关键 字 synchronized， 即 可 保证 两 个 线程 无 法 同时 执行 某 个 对 象 的 同步 方法 。 
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因此 , 第 一 个 子 问题 的 答案 要 视 具 体 情况 而 定 。 如 果 两 个 线程 拥有 该 对 象 的 同一 实例 , 那么 ， 
答案 就 是 否定 的 ， 它 们 不 能 同时 执行 方法 A。 不 过 ， 要 是 这 两 个 线程 拥有 该 对 象 的 不 同 实例 ， 就 
能 同时 执行 方法 A。 

在 概念 上 ， 你 可 以 从 “ 锁 ” 的 角度 来 考虑 答案 。 同 步 方 法 会 对 所 属 对 象 特 定 实 例 的 所 有 同步 
方法 上 锁 ， 从 而 阻止 任何 其 他 线程 执行 那个 实例 的 同步 方法 。 

第 二 个 子 问 题 间 的 是 ，thread2 在 执行 非 同 步 方 法 B 时 ，thread1 能 否 执行 同步 方法 A。 既 然 B 
不 是 同步 方法 ， 在 thread2 执 行 方 法 B 时 ， 也 就 无 从 阻止 thread1 执 行 方 法 。 不 管 thread1 和 thread2 是 
否 拥有 该 对 象 的 同一 实例 ， 这 一 点 都 成 立 。 

说 到 底 ， 此 题 强调 的 关键 概念 是 ,那个 对 象 的 每 个 实例 只 能 执行 一 个 同步 方法 。 其 他 线程 可 
以 执行 该 实例 的 非 同 步 方法 , 或者， 它们 可 以 执行 该 对 象 不 同 实例 的 任意 方法 。 


9.17 ”中 等 难题 























17.1 编写 一 个 函数 ， 不 用 临时 变量 ， 直 接 交 换 两 个 数 。( 第 104 页 ) 


解法 
这 是 个 经 典 面 试题 ， 也 相当 直接 。 我 们 将 用 ae 表示 a 的 初始 值 ，be 表 示 b 的 初始 值 ， 用 diff 表 
示 ae。 - be 的 值 。 
让 我 们 将 a > b 的 情形 绘制 在 数 轴 上 。 
| | | 


| | diff | 
0 b a 


8 a 


首先 ,将 a 设 为 diff， 即 上 面 数 轴 的 右边 那 一 段 。 然 后 ，b 加 上 diff ( 并 将 结果 保存 在 b 中 )， 
就 可 得 到 as。 至 此 ， 我 们 得 到 b = as 和 a = diff。 最 后 ， 只 需 将 b 设 为 a。- diff， 也 就 是 bp - a。 
下 面 是 具体 的 实现 代码 。 














public static void swap(int a, int b) { 
// Wa = 9、b = 4 为 例 

5 

9 


a- D， 
全 二 日 3 
b 


十 


9 吕 
IE 
9 5 避 
庆 
Mo te ss 
uw 
[i 


/a 
/ b 
/a 


汗 


System.out.println(“a: ”+ a); 
System.out.println(“b: ”+ b); 


1 
2 
3 
4 
5 
6 
7 
8 

9 } 

我 们 还 可 以 用 位 操作 实现 类 似 的 解法 , 这 种 解法 的 优点 在 于 它 适用 的 数据 类 型 更 多 , 不 仅 限 
于 整数 。 


1 public static void swap_opt(int a, int b) { 
2 // Wa = 161 (二 进 制 ) 和 b = 116 为 例 
3 a = a^b; // a = 161^116 = 611 
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b 
a 


a^b; // b = 611^116 
a^b; // a = 611^161 


161 
116 


System.out.println(“a: ”+ a); 
System.out.println(“b: ”+ b); 





这 段 代码 使 用 了 蜡 或 操作 ,要 了 解 个 中 细节 , 最 简单 的 方法 就 是 看 看 两 个 比特 位 p 和 q 的 情况 ， 
一 探究 部。 这 里 会 用 pe 和 qe 表示 初始 值 。 

如 能 正确 交换 两 个 比特 位 ， 整 个 操作 就 能 正确 无 误 地 进行 。 下 面 将 逐 行 解析 交换 过 程 。 
pe^qe /* 若 pe = qe 则 为 6， 若 pe != qe 则 为 */ 
p^qe。  /* 等 于 pe 的 值 */ 
p^q ”/* 等 于 qe 的 值 */ 

第 1 行 执行 操作 p = pe^qe， 若 pe = qe 则 结果 为 0; 若 pe! = qe 则 为 1。 

第 2 行 执行 q = p^qe， 可 以 就 p 为 0 和 1 两 种 可 能 的 值 进行 检查 。 最 终 目 的 是 要 交换 p 和 q 的 初始 
值 ， 我 们 和 希望 这 个 操作 返回 pe 的 值 。 
口 若 p = 6: 则 pe。 = qe， 因 此 ， 我 们 需要 该 操作 返回 pe 或 qe。 任 意 值 与 6 异 或 都 会 返回 初始 
值 ， 由 此 可 知 该 操作 会 正确 返回 q。( 或 pe )。 
口 若 p = 1: 则 pe! = qe。 我 们 希望 该 操作 ， 当 qe 为 0 时 返回 1，pe 为 1 时 返回 0。 这 正 是 将 任 

意 值 与 1 执行 异 或 操作 的 结果 。 

第 3 行 执行 p = p^q， 再 次 检查 p 为 0 和 1 两 种 值 的 情况 ， 目 的 是 返回 qe。 注 意 ，q 现 在 等 于 pe， 
因此 其 实 是 在 执行 p^pe。 
口 若 p = 6: 由 于 ps。 = qe， 我 们 希望 该 操作 返回 pe 或 gg， 不 论 哪 一 个 都 可 以 。 执 行 e^pe 会 返 
回 pe， 等 于 qe。 
口 若 p = 1: 该 操作 其 实 是 在 执行 1*pe。 这 会 翻转 pe 的 值 ， 而 这 正 是 我 们 想 要 的 ， 因 为 pe! = qe。 

至 此 ， 我们 已 将 p 设 为 qg。，q 设 为 pe。 综 上 ， 上 述 操作 会 正确 交换 两 个 比特 位 ， 因 此 ， 就 能 正 
确 交 换 整 个 整数 。 


17.2 ”设计 一 个 算法 ， 判 断 玩 家 是 否 赢 了 井 字 游 戏 。( 第 104 页 ) 


解法 

乍 一 看 ,可 能 会 觉得 此 题 真 的 很 简单 ,不 就 是 直接 检查 井 字 棋盘 ,这 会 有 多 难 呢 ? 细 一 想 ,此 
题 还 是 有 点 复 妈 的 ， 而 且 没有 唯一 的 “完美 ”答案 。 根 据 你 的 喜好 不 同 ， 会 有 不 一 样 的 最 佳 解 法 。 

解决 此 题 ， 有 几 个 重要 的 设计 决策 需要 考虑 。 

(1) haswon 只 会 调用 一 次 还 是 很 多 次 〈 比 如 ， 放 在 网 站 上 的 井 字 游戏 ) ?如 果 管 案 是 后 者 ， 
我 们 可 能 会 增加 一 些 预 处 理 ， 以 优化 haswon 的 运行 时 间 。 

(2) 并 字 游 戏 通 常 是 3x3 棋 盘 。 我 们 只 是 针对 3x3 大 小 的 棋盘 进行 设计 ， 还 是 要 实现 一 个 NxN 
的 解法 ? 

(3) 对 于 程序 大 小 、 执 行 速度 和 代码 清晰 度 ， 一 般 如 何 区 分 它们 的 优先 级 呢 ? 记 住 ， 最 高 效 
的 代码 不 一 定 是 最 好 的 。 代 码 是 否 容 易 理 解 、 维 护 也 很 重要 。 





















































7 
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解法 1， 如果 hasWon 会 被 调用 很 多 次 
总 共 只 有 3”， 大 约 20 000 种 井 字 游戏 棋盘 (假设 为 3x3 的 棋盘 )。 因 此 ， 用 一 个 int 就 能 表示 ， 
其 中 每 个 数位 代表 棋盘 中 的 一 格 (0 为 空 、1 为 红 、 2 为 蓝 )。 我 们 会 事先 设 定好 一 个 散 列 表 或 数组 ， 




















将 所 有 可 能 的 棋盘 作为 键 ， 值 则 代表 谁 赢 了 。 这 么 一 来 ，haswon 函 数 就 很 简单 了 : 
1 public int hasWon(int board) { 
2 return winnerHashtable[board]; 
3 } 


要 将 一 个 棋盘 ( 以 字符 数组 表示 ) 转 成 一 个 int ， 可 以 运用 “3 进位 ”表示 法 ， 每 个 棋盘 


可 表示 为 3%vo + 


3 V1 十 33 十... 十 3ivs， 若 格子 为 空 则 vj 为 90， 格子 为 蓝 色 则 vj 为 1， 格 子 为 红色 





则 yj 为 2。 





public static int convertBoardToInt(char[][] board) { 
int factor = 1; 
int sum = @; 


for (int j = 6; j < board[i].length; j++) { 


1 
2 
3 
4 for (int i = 6; i < board.length; i++) { 
5 
6 
7 
8 


int v = 0; 
if (board[i][j] == ‘x’”) { 
VvV=1; 
9 } else if (board[i][j] == “0”) { 
16 V = 2; 
11 } 
12 sum += V * factor; 
13 factor *= 3; 
14 } 
15 } 
16 return sum; 
17 } 





至 此 ， 要 判断 谁 是 赢家 ， 只 需 查 询 散 列 表 即 可 。 
当然 , 如果 每 次 判断 谁 赢 了 都 要 将 棋盘 转 成 这 种 格式 ,那么 跟 其 他 解法 相 比 ， 其 实 并 没有 节 
省 多 少时 间 。 但 是 ， 如 果 一 开始 就 以 这 种 格式 存储 棋盘 那么， 查询 操作 将 会 非常 有 效率 。 


解法 2: 专 为 3x3 棋盘 设计 
如 果 只 想 为 3x3 棋 盘 设 计 一 种 解法 ， 代 码 就 会 比较 简短 上 且 简 单 。 复 杂 的 地 方 只 剩 下 如 何 写 得 


























清晰 而 有 条 理 ， 并 且 不 要 写 出 太 多 重复 代码 。 
1 Piece hasWon1i(Piece[][] board) { 
2 for (int i = 6; i «< board.length; i++) { 
3 /* 检查 行 */ 
4 if (board[i][6] != Piece.Empty && 
5 board[i][8] == board[i][1] && 
6 board[i][8] == board[i][2]) { 
7 return board[i][e]; 
8 } 
9 
16 /* 检查 列 */ 
4 if (board[6][i] != Piece.Empty && 


图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 


9.17 ”中 等 难题 309 





12 board[8][i] == board[1][i] && 
13 board[8][i] == board[2][i]) { 
14 return board[8][i]; 

15 } 

16 } 

17 


18  ”/* 检查 对 角 线 */ 
19 if (board[86][6] != Piece.Empty && 


26 board[8][8] == board[1][1] && 
21 board[8][8] == board[2][2]) { 
22 return board[6][6]; 

23 } 

24 


25 /* 检查 逆 对 角 线 */ 
26 if (board[2][8] != Piece.Empty && 


27 board[2][8] == board[1][1] && 
28 board[2][8] == board[6][2]) { 
29 return board[2][6]; 

36 } 

31 return Piece.Empty; 

32 } 


解法 3: 面向 NxN 棋盘 进行 设计 
有 了 3x3 棋 盘 的 实现 代码 ， 自 然 就 会 想到 要 扩展 到 NxN 棋 盘 。 本 书 可 下 载 的 源码 提供 了 另外 
几 种 解法 ， 下 面 是 其 中 一 种 。 




















1 Piece hasWon3(Piece[][] board) { 

之 int N = board.1length 

3 int row = 9; 

4 int col = 0; 

5 

6 /* 检查 行 */ 

7 for (row = 8; row < Ni row++) { 

8 if (board[row][8] != Piece.Empty) { 

9 for (col = 1; col < N; col++) { 

16 if (board[row][col] != board[row][col-1]) break; 
11 

12 if (col == N) return board[row][6]; 
13 } 

14 } 

15 


16 /* 检查 列 */ 
17 for (col = 6;j col < N; col++) { 


18 if (board[8][col] != Piece.Empty) { 

19 for (row = 1; row < N; row++) { 

26 if (board[row][col] != board[row-1][col]) break; 
21 

22 if (row == N) return board[6][col]; 

23 } 

24 } 

25 


26 /* 检查 对 角 线 (左上 到 右 下 ) */ 
27 if (board[8][8] != Piece.Empty) { 
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28 for (row = 1; row < Ni row++) { 

29 if (board[row][row] != board[row-1][row-1]) break; 
36 } 

31 if (row == N) return board[6][68]; 

32 } 

33 


34 /* 检查 对 角 线 (左下 到 右上 ) */ 
35 if (board[N-1][6] != Piece.Empty) { 


36 for (row = 1; row < Ni row++) { 

37 if (board[N-row-1][row] != board[N-row][row-1]) break; 
38 

39 if (row == N) return board[N-1][6]; 

46 } 

41 

42 return Piece.Empty; 

43 } 





不 论 你 的 解法 为 何 ， 此 题 的 算法 并 不 是 太 难 。 重 点 在 于 理解 如 何 写 出 清晰 、 可 维护 的 代码 ， 





而 这 也 正 是 面试 官 想 要 评估 的 地 方 。 


17.3 ”设计 一 个 算法 ， 算 出 "阶乘 有 多 少 个 尾随 零 。( 第 104 页 ) 


解法 
简单 的 做 法 是 先 算出 阶乘 ， 然 后 不 断 地 除 以 10, 数 一 数 有 几 个 尾随 零 (trailing zero )。 但 





这 种 做 法 的 问题 是 , 使 用 int 很 快 就 会 越界 。 为 了 避 开 这 个 限制 ,我 们 可 以 从 数学 上 来 分 析 这 


个 问题 。 


下 面 以 阶乘 191 为 例 进行 说 明 : 

19! = 1*2*3*4*5*6*7]*8*9*1@*11*12*13*14*15*16*17*18*19 

10 倍 数 就 会 形成 尾随 零 ， 而 10 倍 数 又 可 分 解 为 一 组 组 5 倍数 和 2 倍数 。 

例如 ， 在 19! 中 ， 下 列 几 项 会 形成 尾随 零 : 

9 由 E20 6 

因此 , 为 了 算出 尾随 零 的 数量 , 我 们 只 需 计算 有 几 对 5 和 2 倍数 。 不 过 ，2 售 数 始 终 要 比 5 倍数 
最 后 只 要 数 出 5 倍数 就 可 以 了 。 

这 里 有 个 陷阱 , 就 是 15 只 能 算 一 个 5 倍数 ( 因此 会 形成 一 个 尾随 零 ), 而 25 算 两 个 (25 =5*5 )。 
编写 代码 时 ， 相 关 代码 有 两 种 写法 。 

第 一 种 写法 是 迭代 访问 所 有 2 到 n 的 数字 ,计算 每 个 数字 中 有 几 个 5。 


















































1 /* 车 数字 为 5 的 倍数 ， 返回 5 的 几 次 方 
六、 深 5 -> 1, 

3 .并 25-> 2 等 

4 */ 

5 public int factorsOf5(int i) { 
6 int count = 0; 

7 while (i % 5 == 6) { 

8 Count++; 

9 i /= 5; 
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16 } 

11 return count; 
12 } 

13 


14 public int countFactZeros(int num) { 
15 int count = 0; 


16 for (int i = 2; i <= num; i++) { 
17 count += factorsOf5(i); 

18 } 

19 return count; 

26 } 


这 么 写 还 不 赖 , 不 过 , 我 们 还 可 以 做 得 更 有 效率 一 点 : 直接 数 一 数 5 的 因数 。 采 用 这 种 做 法 ， 
我 们 会 先 数 一 数 1 到 "之 间 ， 有 几 个 5 的 倍数 〈 数 量 为 w5 )， 然 后 数 一 数 25 的 倍数 有 几 个 〈m/25 )， 
接着 是 125， 依 此 类 推 。 

要 算出 n 中 有 几 个 m 的 倍数 ， 直 接 将 n 除 以 m 即 可 。 











public int countFactZzeros(int num) { 
int count = 0; 

if (num < 6) { 

return -1; 


for (int i = 5; num / i > 6; i *= 5) { 
count += num / i; 


} 


1 
2 
3 
4 
5 } 
6 
7 
8 
9 return count; 


16 } 

此 题 有 点 像 脑筋 急 转 弯 ， 不 过 ， 还 是 可 以 通过 逻辑 思考 来 解决 (如 上 所 示 )。 只 要 思考 一 下 
到 底 有 哪些 条 件 会 形成 尾随 零 ， 就 能 得 到 解法 。 你 必须 从 一 开始 就 透彻 地 理解 相关 规则 ,才能 正 
确 地 实现 出 来 。 


17.4 ”编写 一 个 方法 ， 找 出 两 个 数字 中 最 大 的 那 一 个 。 不 得 使 用 if-else 或 其 他 比较 运算 符 。 
(第 104 页) 


解法 

max 函 数 的 常见 实现 方式 是 检查 a - 5b 的 正 负 号 。 但 这 里 不 能 使 用 比较 运算 符 检查 正 负 情况 ， 
不 过 我 们 可 以 使 用 乘法 。 

假定 i 代 表 a -5 的 正 负 号 ， 如 果 a 一 b>=0， 则 为 1， 否 则 为 0。 令 gq 为 的 反 数 。 

那么 ， 我 们 可 以 实现 如 下 代码 : 
/* 1 变 8，6 变 1 */ 


public static int flip(int bit) { 
return 1^bit; 


} 


























/* a 为 正则 返回 1，a 为 负 则 返回 8@ */ 
public static int sign(int a) { 


1 
2 
3 
4 
5 
6 
7 
8 return flip((a >> 31) & Ox1); 
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11 public static int getMaxNaive(int a, int b) { 
12 int k = sign(a - b); 

13 int q = flip(k); 

14 returna*k+b*q; 





这 上段 代码 看 似 可 行 ， 实则 不 济 。 要 是 a -bp 溢出 ,这 段 代码 就 行 不 通 。 例如, 假设 a 为 INT_MAX 
- 2，b 为 -15。 此 时 ，a -=- 2 将 大 于 INT_MAX 并 且 会 溢出 ， 最 终 变 为 负 值 。 

运用 同样 的 方法 ,我 们 可 以 实现 此 题 的 解法 ,目标 是 当 a > 5b 时 维持 k 为 1 的 条 件 。 为 此 ， 我 们 
需要 使 用 更 为 复杂 的 逻辑 。 

a 一 b 什 么 时 候 会 溢出 呢 ?” 它 只 会 在 a 为 正 、5b 为 负 时 溢出 , 或 者 ， 反 这 来 也 有 可 能 。 专门 检测 
溢出 条 件 可 能 比较 困难 , 不 过 , 我 们 可 以 检测 a 和 4b 何 时 会 有 不 同 的 正 负 号 。 注 意 ， 如 果 a 和 4b 的 正 
负 号 不 同 ， 就 让 k 等 于 sign(a)。 

具体 逻辑 如 下 : 














1 if a 和 b 的 正 负 号 不 同 : 

2 // 车 a > 6, 则 b < 6 且 k = 1 

3 // 车 a < 6, 则 b>6 且 k = 6 

4 // 因此 ， 不 管 哪 种 情况 ，k = sign(a) 

5 let k = sign(a) 

6 else 

7 let k = sign(a - b) // 这 里 不 再 有 溢出 





上 述 逻 辑 的 实现 代码 如 下 ， 其 中 使 用 了 乘法 而 不 是 if 语 人 句 。 


1 public static int getMax(int a, int b) { 

2 int c=a-b; 

3 

4 int sa = sign(a); // if a >= 08, then 1 else 6 
5 int sb = sign(b); // if b >= 6, then 1 else 6 
6 int sc = sign(c); // 取决 于 a - b 有 没有 溢出 

7 

8 /* 目标 : 定义 k 的 值 ， 若 a > b 则 为 1，a < b 则 为 6 

9 * (车 a = b，k 为 何 值 无 关 紧 要 ) */ 

16 

4 // 若 a 和 b 正 负 号 不 同 ， 则 K = sign(a) 

12 int use_sign of a = Sa ^ sb; 

13 


14 // 若 a 和 b 正 人 负 号 相同 ,， 则 k = sign(a - b) 
15 int use_sign of c = flip(sa ^ sb); 


16 

17 int k = use_ sign of a * sa + use sign of c * sc; 
18 int q = flip(k); // kk 的 反 数 

19 

20 returna*k+b*q; 

2 - 

















意 , 为 清晰 起 见 , 我 们 将 代码 拆 分 成 多 个 方法 和 变量 。 很 显然 , 这 不 是 最 紧凑 或 最 有 效率 
和 但 这 么 写 代码 要 清晰 许多 。 
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17.5“ 珠 现 妙 算 游戏 (The Game of Master Mind ) 的 玩法 如 下 。 

计算 机 有 四 个 槽 , 每 个 槽 放 一 个 球 , 颜色 可 能 是 红色 (R 黄色 (Y 绿色 (G ) 或 蓝 色 (B )。 
例如 ， 计 算 机 可 能 有 RGGB 四 种 ( 槽 1 为 红色 ,， 槽 2、3 为 绿色 , 槽 4 为 赣 色 )。 

作为 用 户 ， 你 试图 猜 出 颜色 组 合 。 打 个 比方 ， 你 可 能 会 猜 YRGB。 

要 是 猜 对 某 个 槽 的 颜色 ， 则 算 一 次 “ 猜 中 ”; 要 是 只 猜 对 颜色 但 槽 位 猜 错 了 ， 则 算 一 次 “ 伪 
猜 中 "。 注 意 ,“ 猜 中 ”不 能 算 入 “ 伪 猜 中 ”。 

举 个 例子 ， 实 际 颜 色 组 合 为 RGBY ， 而 你 猜 的 是 GGRR ， 则 算 一 次 猜 中 ， 一 次 伪 猜 中 。 

给 定 一 个 猜测 和 一 种 颜色 组 合 ， 编 写 一 个 方法 ， 返 回 猜 中 和 伪 猜 中 的 次 数 。( 第 104 页 ) 


解法 

此 题 简单 明了 , 但 令 人 惊讶 的 是 ， 写 代码 时 很 容易 犯 小 错误 。 代 码 写 好 后 ， 你 应 该 对 照 各 种 
测试 用 例 ， 进 行 全 面 彻底 的 检查 。 

编写 代码 时 ,我们 首先 会 构造 一 个 频率 数组 ， 存 放 每 个 字符 在 solution 中 出 现 的 次 数 ， 但 
不 包括 某 个 模 被 “ 猜 中 ”的 次 数 。 然 后 ， 和 迭代 guess 算 出 伪 猜 中 的 次 数 。 





























下 面 是 这 个 算法 的 实现 代码 。 

1 public class Result { 

2 public int hits = @; 

3 public int pseudoHits = 0@; 

4 

5 public String toSstring() { 

6 return “(“ + hits + “, “ + pseudoHits + ")”; 
7 } 

8 } 


9 
16 public int code(char c) { 
11 switch (c) { 


12 case “B? : 

13 return ©; 
14 Case “G? : 

15 return 1; 
16 Case R’: 

17 return 2; 
18 Case Y’: 

19 return 3; 
20 default: 

21 return -1; 
22 } 

23 } 

24 

25 public static int MAX_ COLORS = 4; 
26 


27 public Result estimate(String guess, String solution) { 
28 if (guess.length() != solution.length()) return null; 


36 Result res = new Result(); 
31 int[] frequencies = new int[MAX_COLORS]; 
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33 /* 计算 猜 中 次 数 ， 构 造 频率 表 */ 
34 for (int i = 6; i < guess.length(); i++) { 


35 if (guess.charAt(i) == solution.charAt(i)) { 
36 res.hits++; 

37 } else { 

38 /* 只 有 不 是 猜 中 的 情况 下 ， 才 增加 频率 表 
39 * (将 用 于 伪 猜 中 ) 。 若 是 猜 中 ， 那 么 ， 
46 * 该 槽 位 已 被 “使 用 ”*#/ 

41 int code = code(solution.charAt(i)); 
42 frequencies[code]++; 

43 } 

44 } 

45 


46 /* 计算 伪 猜 中 */ 
47 for (int i = 6; i < guess.length(); i++) { 


48 int code = code(guess.charAt(i)); 

49 if (code >= 6 && frequencies[code] > 6 && 

56 guess.charAt(i) != solution.charAt(i)) { 
51 res.pseudoHits++; 

52 frequencies[code]--; 

53 } 

54 } 

55 return res; 

56 } 





注意 ， 问 题 所 需 的 算法 越 简单 ， 写 出 清晰 、 正 确 的 代码 就 越 显 重要 。 在 上 面 的 例子 中 ,我 
们 提取 代码 专门 写 了 个 code(char c) 方 法 ， 并 创建 了 一 个 Result 类 来 保存 结果 ， 而 非 只 是 打印 
显示 。 


17.6 ”给 定 一 个 整数 数组 ， 编 写 一 个 国 数 ， 找 出 索引 m 和 z*， 只 要 将 m 和 xz 之 间 的 元 素 排 好 
序 ， 整 个 数组 就 是 有 序 的 。 注 意 : n - 光 越 小 越 好 ， 也 就 是 说 ， 找 出 符合 条 件 的 最 短 序列 。( 第 
104 页 ) 


解法 
开始 解 题 之 前 ， 让 我 们 先 确认 一 下 答案 会 是 什么 样 的 。 如 果 要 找 的 是 两 个 索引 ， 这 表明 数组 
中 间 有 一 段 有 竺 排序， 其 中 数组 开头 和 来 尾部 分 是 排 好 序 的 。 
现在 ， 我 们 借用 下 面 的 例子 来 解决 此 题 : 
1，2，4，7，16，11，7，12，6，7，16，18，19 
首先 映 入 脑海 的 想法 可 能 是 , 直接 找 出 位 于 开头 的 最 长 递增 子 序 列 , 以 及 位 于 末尾 的 最 长 递 
增 子 序 列 。 
左边 : 1，2，4，7，16，11 
中 间 : 7，12 
右边 : 6，7，16，18，19 
很 容易 就 能 找 出 这 些 子 序列 ， 只 需 从 数组 最 左边 和 最 右边 开始 ,向 中 间 查 找 递 增 子 序列 。 一 
且 发 现 有 元 素 大 小 顺序 不 对 ， 那 就 是 找到 了 递增 /递减 子 序列 的 两 头 。 
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但 是 , 为 了 解决 这 个 问题 , 还 需要 对 数组 中 间 部 分 进行 排序 ， 只 要 将 中 间 部 分 排 好 序 ， 数组 
所 有 元 素 便 是 有 序 的 。 具 体 来 说 ， 就 是 以 下 判断 条 件 必须 为 真 : 


/* 左边 (left) 所 有 元 素 都 要 小 于 中 间 (middle) 的 所 有 元 素 */ 
min(middle) > end(left) 























/* 中 间 (middle) 所 有 元 素 都 要 小 于 右边 (right) 的 所 有 元 素 */ 
max(middle) < start(right) 


或 者 ， 换 句 话说 ， 对 于 所 有 元 素 : 
left «< middle < right 

实际 上 ， 上 例 的 这 个 条 件 绝 不 可 能 成 立 。 根 据 定义 ,中 间 部 分 的 元 素 是 无 序 的 。 而 在 上 面 的 
例子 中 ，1left.end > middle.start 且 middle.end > right.start 一 定 成 立 。 这 样 一 来 ， 只 排 
序 中 间 部 分 并 不 能 让 整个 数组 有 序 。 

不 过 ,我 们 还 可 以 缩减 左边 和 右边 的 子 序列 ， 直 到 先前 的 条 件 成 立 为 止 。 

令 min 等 于 min(middle) ，max 等 于 max(middle)。 

对 左边 部 分 ,我们 先 从 这 个 子 序列 的 末尾 开始 ( 值 为 11， 索 引 为 5 )， 并 向 左 移动 ,直至 找到 
元 素 索 引 i 使 得 array[i] < min; 找到 后 只 需 排序 中 间 部 分 ， 就 能 让 数组 的 那 部 分 有 序 。 

然后 , 对 右边 部 分 进行 类 似 操 作 , 此 时 max 等 于 12。 我 们 先 从 右边 子 序列 的 起 始 元 素 ( 值 为 6 ) 
开始 ， 并 向 右 移 动 ， 将 中 间 部 分 的 最 大 值 12 依 次 与 6、7、16 比 较 。 找 到 16 时 ， 就 能 确定 在 16 的 右 
边 已 经 没有 元 素 比 12 小 了 ( 因为 右边 是 递增 子 序列 )。 至 此 ， 对 数组 中 间 部 分 进行 排序 ， 以 使 整 



































个 数组 都 是 有 序 的 。 
下 面 是 这 个 算法 的 实现 代码 。 
1 int findEndofLeftSubsequence(int[] array) { 
2 for (int i = 1; i < array.length; i++) { 
3 if (array[i] < array[i - 1]) return i - 1; 
4 
5 return array.length - 1; 
6 } 
了 
8 int findstartOofRightSubsequence(int[] array) { 
9 for (int i = array.length - 2; i >= 6; i--){ 
16 if (array[i] > array[i + 1]) return i + 1; 
11 
12 return 6 
13 } 
14 
15 int shrinkLeft(int[] array, int min_index, int start) { 
16 int comp = array[min_index]; 
17 for (int i = start - 1; i >= 6; i--){ 
18 if (array[i] <= comp) return i + 1; 
19 
26 return ©; 
21 } 
22 


23 int shrinkRight(int[] array, int max_index, int start) { 
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24 int comp = array[max_index]; 

25 for (int i = start; i < array.length; i++) { 
26 if (array[i] >= comp) return i - 1; 

27 } 

28 return array.length - 1; 

29 } 

30 


31 void findUnsortedSequence(int[] array) { 

32 /* 找 出 左 子 序列 */ 

33 int end_left = findEndofLeftSubsequence(array); 

34 

35 /* 找 出 右 子 序列 */ 

36 int start right = findSstartOofRightSubsequence(array ) ; 


37 

38 /* 找 出 中 间 部 分 的 最 小 值 和 最 大 值 */ 

39 int min_index = end_ left + 1; 

46 if (min_index >= array.length) return; // 已 排序 
41 

42 int max_index = start right - 1; 

43 for (int i = end left; i <= start right; i++) { 
44 if (array[i] < array[min_index]) min_index = i; 
45 if (array[i] > array[max_index]) max_index = i; 
46 } 

47 


48 /* 向 左 移动 ， 直 到 小 于 array[min_index] */ 

49 int left_index = shrinkLeft(array, min_index, end_left); 

56 

51 /* 向 右 移动 ， 直 到 大 于 array[max_index] */ 

52 int right_ index = shrinkRight(array, max_index, start_ right); 
53 

54 System.out.println(left_index + “ “ + right_ index); 

55 } 


注意 , 在 上 面 的 解法 中 , 我们 还 创建 了 不 少 方法 。 虽然 也 可 以 把 所 有 代码 一 股 脑 儿 塞 进 一 个 
方法 , 但 这 样 一 来 ,代码 理解 、 维 护 和 测试 起 来 就 要 难得 多 。 在 面试 中 写 代码 时 ,你 应 该 优先 考 
虑 这 几 点 。 


17.7 ”给 定 一 个 整数 ， 打 印 该 整数 的 英文 描述 (例如 “One Thousand, Two Hundred- 
Thirty Four”)。( 第 104 页 ) 


解法 

此 题 并 不 太 难 ， 反 倒 有 点 乏味 。 关 键 在 于 解 题 的 过 程 和 组 织 ， 并 确定 你 有 完善 的 测试 用 例 。 

举 个 例子 ， 在 转换 19 323 984 时 ， 我 们 可 以 考虑 分 段 处 理 ， 每 三 位 转换 一 次 ， 并 在 适当 的 地 
方 插入 “thousand”( 千 ) 和 “million”( 百 万 )。 也 即 ， 


convert(19 323 984) = convert(19) + “ million ”+ 
convert(323) + “ 上 thousand ”+ 
convert(984) 


下 面 是 该 算法 的 实现 代码 。 
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1 public String[] digits = {“One”, “Two”, “Three”, “Four”, “Five”, 

2 “Six”, “Seven”, “Eight”, “Nine”}; 

3 public String[] teens = {“Eleven”, “Twelve”, “Thirteen”, 

4 “Fourteen”, “Fifteen”, “Sixteen”, “Seventeen”, “Eighteen”, 

5 “Nineteen”}; 

6 public static String[] tens = {“Ten”, “Twenty”, “Thirty”, “Forty”, 
7 “Fifty”, “Sixty”, “Seventy”, “Eighty”, “Ninety”}; 

8 public static String[] bigs = {"“”, “Thousand”, “Million”}; 


16 public static String numToString(int number) { 
11 if (number == 0) { 


12 return “Zero”; 

13 } else if (number < 0) { 

14 return “Negative “ + numToString(-1 * number); 
15 } 

16 


17 int count = 0; 
18 String str = “; 


19 

26 while (number > 6) { 

21 if (number % 1666 != 6) { 
22 str = numToSstring166(number % 1666) + bigs[count] + 
23 “72”+ str; 

24 } 

25 number /= 186808; 

26 count++; 

27 } 

28 

29 return str; 

30 } 

31 


32 public static String numToSstring166(int number) { 
33 String str = “”; 


35 /* 转换 百 位 数 的 地 方 */ 
36 if (number >= 166) { 


37 str += digits[number / 166 - 1] + “Hundred ”; 
38 number %= 160; 

39 } 

40 


41 /* 转换 十 位 数 的 地 方 */ 
42 if (number >= 11 && number <= 19) { 


43 return str + teens[number - 11] + “ ”; 
44 } else if (number == 16 || number >= 26) { 
45 str += tens[number / 106 - 1] + “了 

46 number %= 108; 

47 } 

48 


49 /* 转换 个 位 数 的 地 方 */ 
56 if (number >= 1 && number <= 9) { 


5 了 str += digits[number - 1] + “ ”; 
52 } 

S53 

54 return str; 

55 } 


图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 





318 第 9 章 解 题 技 巧 








处 理 这 类 问题 的 关键 在 于 ， 因 为 有 很 多 特殊 情况 ， 所 以 要 确保 考虑 到 所 有 特殊 情况 。 


17.8 ”给 定 一 个 整数 数组 (有 正 数 有 负数 )， 找 出 总 和 最 大 的 连续 数列 ， 并 返回 总 和 。( 第 
104 页) 


解法 
此 题 难度 不 小 ， 但 又 极为 常见 。 接 下 来 ， 我 们 会 通过 下 面 的 例子 来 解 题 : 
2 3 -8 -1 2 4 -2 3 


如 有 果 把 上 面 的 数组 看 作 是 正 数 数列 和 负数 数量 交替 出 现 , 我 们 会 发 现 , 答案 绝 不 会 只 包含 某 
负数 子 数列 或 正 数 子 数列 的 一 部 分 。 何 以 见得 ”只 包含 某 负数 子 数列 的 一 部 分 ,将 使 得 总 和 过 小 ， 
我 们 应 该 排除 整个 负数 数列 才 对 。 同 样 地 ， 只 包含 正 数 子 数列 的 一 部 分 也 会 显得 很 怪 ， 因 为 若 包 
含 整个 子 数列 ， 总 和 就 能 变 得 更 大 。 

为 了 构思 出 算法 , 我 们 可 以 把 数组 看 作 一 个 正 负 数 交 错 出 现 的 数列 。 每 个 数字 代表 正 数 子 数 
列 的 总 和 ， 或 负数 子 数列 的 总 和 。 对 于 上 面 的 数组 ， 简 化 后 如 下 : 

5 -9 6 -2 3 

我 们 无 法 从 中 立即 窥 得 很 棒 的 算法 , 不 过 , 它 确实 可 以 帮助 我 们 更 好 地 理解 手头 正在 处 理 的 
问题 。 

考虑 上 面 的 数组 。 把 {5，-9} 视 作 子 数列 说 得 通 吗 ?不 ， 这 两 个 数字 的 总 和 为 -4， 所 以 最 好 
两 个 数字 都 不 要 ， 或 者 考虑 只 包含 子 数列 {5}， 只 有 一 个 元 素 。 

什么 情况 下 需要 在 子 数 列 中 包含 负数 呢 ? 只 有 当 它 能 将 两 个 正 子 数列 拼接 在 一 起 , 并 且 两 者 
加 起 来 大 于 这 个 负数 的 时 候 。 

我 们 可 以 一 步 一 步 地 找 出 答案 ， 先 从 数组 的 第 一 个 元 素 开始 。 

首先 看 到 5， 这 是 到 目前 为 止 最 大 的 总 和 。 我 们 将 maxsum 设 为 5， 并 将 sum 设 为 5。 接 着 ， 考 
虑 -9, 将 它 与 sum 相 加 会 得 到 负 值 。 将 子 数列 从 5 延伸 到 -9 并 没有 意义 ( 只 会 将 子 数列 缩减 为 -4 )， 
因此 我 们 会 重 置 sum 的 值 。 

现在 看 到 6， 这 个 子 数 列 比 5 大 ， 因 此 更 新 maxsum 和 sum。 

接着 来 看 -2， 与 6 相 加 ，sum 设 为 4。 由 于 总 和 仍 会 变 大 〈 与 其 他 部 分 连接 时 ， 会 有 更 长 的 数 
列 )， 我 们 有 可 能 想 把 tf6，-2} 纳 入 最 长 子 数列 ， 因 此 更 新 sum， 但 不 更 新 maxsum。 

最 后 看 到 3，3 加 上 sum (4 ) 结果 为 7， 更 新 maxsum， 最 后 得 到 最 长 子 数列 为 {6，-2，3}。 

推 而 广 之 ， 对 于 完全 展开 的 数组 而 言 ， 处 理 逻 辑 是 一 样 的 。 下 面 是 该 算法 的 实现 代码 。 

1 public static int getMaxSum(int[] a) { 

2 int maxsum = 0@; 
int sum = @; 
for (int i = 6;j i < a.length; i++) { 

sum += a[i]; 


if (maxsum < sum) { 
maxsum = sum; 
















































































NOm 人 ww 


图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 


9.17 中 等 难题 319 





8 } else if (sum < 6) { 
9 sum = 0; 

16 } 

11 } 

12 return maxsum; 

13 } 





如 果 整 个 数组 都 是 负数 ， 怎 么 样 才 是 正确 的 行为 ”看 看 这 个 简单 的 数组 : {-3，-16，-5}， 
以 下 答案 每 个 都 说 得 通 : 

(1) -3( 假设 子 数列 不 能 为 空 ); 

(2) 0 ( 子 数 列 长 度 为 零 ); 

(3) MINIMUM INT ( 视 为 错误 情况 )。 

我 们 会 选择 第 二 个 ( maxsum = 8 )， 但 其 实 并 没有 所 谓 的 “正确 ”答案 。 这 一 点 可 以 跟 面 试 
官 好 好 讨论 一 番 ， 这 样 也 能 展示 出 你 是 个 注重 细节 的 人 。 


17.9 设计 一 个 方法 ， 找 出 任意 指定 单词 在 一 本 书 中 的 出 现 频率 。( 第 104 页 ) 


解法 
面对面 试 官 ， 该 问 的 第 一 个 问题 是 ， 这 个 操作 是 执行 一 次 还 是 不 断 被 执行 。 也 就 是 说 ， 你 是 
只 要 找 出 “dog” 的 频率 ， 还 是 想 找 出 “dog”， 接 着 是 “cat”、“mouse”， 等 等 ? 


1. 解法 : 单 次 查询 
在 这 种 情况 下 , 我 们 会 直接 逐 字 逐 名 地 扫描 整 本 书 , 数 一 数 某 个 单词 出 现 的 次 数 , 用 时 O(n)。 
可 以 确定 这 是 最 短 用 时 ， 因 为 不 管 怎么 样 ， 我 们 必须 查看 过 书 中 的 每 个 单词 。 


2. 解法 :重复 查询 

如 果 是 要 重复 执行 查询 操作 ,那么 , 或许 值得 我 们 多 花 些 时 间 ， 多 分 配 内 存 ， 对 全 书 进行 预 
人 处理 。 我 们 可 以 构造 一 个 散 列表 , 将 单词 映射 到 该 单词 的 出 现 频 率 , 这 么 一 来 ,任意 单词 的 频率 
都 能 在 O(1) 时 间 内 找到 。 具 体 实现 代码 如 下 。 



























































1 Hashtable<String, Integer> setupDictionary(String[] book) { 
2 Hashtable<String, Integer> table = 

3 new Hashtablex<String, Integer>(); 

4 for (String word : book) { 

5 word = word.toLowerCase(); 

6 if (word.trim() != “») { 

7 if (!table.containsKey(word)) { 

8 table.put(word, 8); 

9 } 

16 table.put(word, table.get(word) + 1); 
11 } 

12 } 

13 return table; 

14 } 

15 





16 int getFrequency(Hashtable<String, Integer> table, String word) { 
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17 if (table == null || word == null) return -1; 
18 word = word.toLowerCase(); 
19 if (table.containsKey(word)) { 
26 return table.get(word); 
21 
22 return 6 
23 } 
注意 ， 相 对 而 言 ， 这 类 问题 还 是 比较 容易 的 。 因 此 ， 面 试 官 会 更 看 重 你 的 心思 有 多 续 密 ， 有 


没有 检查 错误 条 件 ? 


17.10 ”XML 非常 见长， 你 找到 一 种 编码 方式 ， 可 将 每 个 标签 对 应 为 预先 定义 好 的 整数 值 ， 
该 编码 方式 的 语法 如 下 : 


Element --> Tag Attributes END Children END 
Attribute --> Tag Value 

END -->0 

Tag --> 对 应 到 某 个 预先 定义 好 的 整数 值 
Value --> 字符 串 值 END 


例如 ， 下 列 XML 会 被 转换 压缩 成 下 面 的 字符 串 (假定 对 应 关系 为 family -> 1、person 


-> 2、firstName -> 3、lastName -> 4、state -> 5)。 


<family lastName=“McDowell” state=“CA”> 
<person firstName=“Gayle”>Some Message</persony> 
</family> 


变 为 : 
1 4 McDowell 5 CA 68 2 3 Gayle 6 Some Message 6 6. 


编写 代码 ， 打 EjXML 元 素 编码 后 的 版 本 ( 传 入 Element 和 Attribute 对 象 )。( 第 105 页 ) 


解法 

由 题目 可 知 ， 元 素 会 以 Element 和 Attribute 作 为 参数 传人 ， 因 此 具体 代码 相当 简单 ， 可 以 
运用 类 似 树 状 结构 的 做 法 实现 。 

我 们 会 不 断 对 XML 结 构 的 各 个 部 分 调用 encode(), 根据 XML 元 素 的 类 型 , 处 理 方式 稍 有 不 同 。 








1 public static void encode(Element root, StringBuffer sb) { 
2 encode(root.getNameCode(), sb); 

3 for (Attribute a : root.attributes) { 

4 encode(a, sb); 

5 } 

6 encode(“6@”, sb); 

7 if (root.value != null && root.value != “») { 
8 encode(root.value, sb); 

9 } else { 

16 for (Element e : root.children) { 

11 encode(e, sb); 

12 } 

13 

14 encode(“@”, sb); 

15 } 
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17 public static void encode(String v, StringBuffer sb) { 
18 sb.append(v); 

19 sb.append(® “); 

20 } 


22 public static void encode(Attribute attr, StringBuffer sb) { 
23 encode(attr.getTagCode(), sb); 

24 encode(attr.value, sb); 

25 } 


27 public static String encodeToString(Element root) { 
28 StringBuffer sb = new StringBuffer(); 


29 encode(root, sb); 
36 return sb.toString(); 
31 } 


请 留意 第 17 行 代码 ， 有 个 负责 处 理 字符 串 的 简单 方法 encode。 这 个 方法 似乎 有 点 画蛇添足 ， 
它 无 非 就 是 插入 字符 串 并 附加 一 个 空格 。 不 过 , 使 用 这 个 方法 有 个 好 处 ， 可 以 确保 每 个 元 素 之 间 
都 有 空格 。 否 则 ， 很 可 能 就 会 忘记 附加 空 日 符 从 而 打 乱 编码 。 


17.11 给 定 rand5() ， 实 现 一 个 方法 rand7()。 也 即 ， 给 定 一 个 产生 0 到 4 ( 含 ) 随机 数 
的 方法 ， 编 写 一 个 产生 0 到 6 ( 含 ) 随机 数 的 方法 。( 第 105 页 ) 

解法 

这 个 函数 要 正确 实现 ， 则 返回 0 到 6 之 间 的 值 ， 每 个 值 的 概率 必须 为 1/7。 

1. 第 一 次 尝试 (调用 次 数 固定 ) 

第 一 次 尝试 时 ， 我 们 可 能 会 想 产 生出 0 到 9 之 间 的 值 ， 然 后 再 除 以 7 取 余 数 。 代 码 大 致 如 下 : 
int rand7() { 

int v = rand5() + rand5(); 


return v % 7; 


} 

可 惜 的 是 ， 上 面 的 代码 无 法 以 相同 的 概率 产生 所 有 值 。 分 析 一 下 每 次 调用 rand5() 返 回 的 结 
果 与 rand7() 函 数 返 回 值 的 对 应 关系 ， 就 能 确认 这 一 点 。 

因为 每 一 行 会 调用 两 次 rand5(), 每 次 调用 返回 不 同 值 的 概率 为 /5， 所 以 ,每 一 行 出 现 的 概 
率 为 /25。 数 一 数 每 个 数字 出 现 的 次 数 ， 就 会 发 现 这 个 rand7() 函 数 以 5/25 的 概率 返回 4， 而 返回 
0 的 概率 为 3/25。 也 就 是 说 ， 这 个 函数 与 题目 要 求 不 符 ， 返 回 各 种 结果 的 概率 并 非 1/7。 

现在 设想 一 下 ， 寿 我 们 要 修改 上 面 的 函数 加 上 一 条 if 语 句 ， 并 修改 常数 乘 数 或 再 插入 一 个 
rand5() 调 用 ， 同 样 会 产生 一 张 类 似 的 表格 ， 而 每 一 行 组 合 出 现 的 概率 将 是 1/5*， 其 中 为 那 一 行 
调用 rand5() 的 次 数 。 不 同行 调用 rand5() 的 次 数 可 能 不 同 。 

最 终 ，rand7() 函 数 返 回 结果 的 概率 ， 比 如 6， 为 所 有 结果 为 6 的 行 的 概率 总 和 ， 也 就 是 : 
P(rand7() = 6)= 1/3 +1/3 + + 1/5” 


为 了 保证 函数 正确 实现 ， 这 个 概率 必须 等 于 1/7。 
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但 这 又 不 可 能 ， 因 为 5 和 7 互 质 ，5 倒 数 的 指数 级 数 不 可 能 得 到 1/7。 

难道 此 题 无 解 吗 ? 并 非 如 此 。 严 格 地 说 , 这 意味 着 , rand5() 调 用 组 合 的 结果 若 能 得 到 rand7() 
的 某 个 特定 值 ， 只 要 能 列 出 来 ， 该 函数 就 不 会 返回 均匀 分 布 的 结果 。 

我 们 还 是 有 办 法 解 出 此 题 的 ， 只 不 过 必须 使 用 while 循 环 ， 另 外 请 注意 ， 我 们 无 法 确定 返回 
一 个 结果 要 经 过 儿 次 循环 。 























2. 第 二 次 尝试 (调用 次 数 不 定 ) 

只 要 能 使 用 while 循 环 ， 工 作 就 会 变 得 简单 许多 。 我 们 只 需 产 生出 一 个 范围 的 数值 ， 且 每 个 
数值 出 现 的 概率 相同 ( 且 这 个 范围 至 少 要 有 7 个 元 素 )。 如 果 能 做 到 这 一 点 , 就 可 以 舍弃 后 面 大 于 
7 的 倍数 的 部 分 ， 然 后 将 余下 元 素 除 以 7 取 余 数 。 由 此 将 得 到 范围 0 到 6 的 值 ， 且 每 个 值 出 现 的 概率 
相等 。 

下 面 的 代码 会 通过 5 * rand5() + rand5() 产 生 范 围 0 到 24。 然 后 ， 含 弃 21 和 24 之 间 的 数值 ， 
否则 rand7() 返 回 0 到 3 的 值 就 会 偏 多 , 最 后 除 以 7 取 余 数 , 得 到 范围 0 到 6 的 数值 , 每 个 值 出 现 的 概 
率 相同 。 

注意 ， 这 种 做 法 需要 舍弃 一 些 值 ， 因 此 不 确定 返回 一 个 值 要 调用 几 次 rand5()， 这 就 是 所 谓 
的 调用 次 数 不 定 。 





























1 public static int rand7() { 

2 while (true) { 

3 int num = 5 * rand5() + rand5(); 
4 if (num < 21) { 

5 return num % 7; 

6 } 

7 } 

8 } 
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注意 ， 执 行 5 * rand5() + rand5() 正 好 只 提供 了 一 种 方式 来 取得 范围 0 到 24 之 间 的 每 个 数 
值 ， 这 就 确保 了 每 个 值 出 现 的 概率 相同 。 

可 以 换个 做 法 执行 2 * rand5() + rand5() 吗 ? 不 行 ， 因 为 这 些 值 不 是 均匀 分 布 的 。 例 如 ， 
取得 6 有 两 种 方式 (6=2*1+4 和 6 = 2*2+2 )， 而 取得 0 ( 6=2*6+6 ) 则 只 有 一 种 方式 ， 在 范围 里 的 
值 出 现 概率 不 等 。 

还 有 一 种 做 法 就 是 使 用 2 * rand5()， 这 样 也 能 得 到 均匀 分 布 的 值 ， 但 要 复杂 得 多 。 代 码 如 下 : 








1 public int rand7() { 

2 while (true) { 

3 int r1 = 2 * rand5(); /* 8 和 9 之 间 的 偶数 */ 
4 int r2 = rand5(); /* 之 后 会 用 来 产生 6 或 1 */ 
5 if (r2 != 4) { /* r2 有 多 余 的 偶数 ， 含 弃 之 */ 
6 int rand1 = r2 % 2; /* 产生 6 或 1 */ 
7 
8 





int num = rl + rand1; /* 将 会 在 范围 6 到 9 之 间 */ 
if (num < 7) { 

9 return num; 

16 } 


12 } 


事实 上 , 我 们 可 以 使 用 的 范围 是 无 限 的。 关键 在 于 确保 该 范围 足够 大 , 且 范 围 内 所 有 值 出 现 
的 概率 相同 。 


17.12 ”设计 一 个 算法 ， 找 出 数组 中 两 数 之 和 为 指定 值 的 所 有 整数 对 。( 第 105 页 ) 

解法 

此 题 有 两 种 解法 ,至 于 哪 一 种 “比较 好 ”， 取 决 于 你 在 时 间 效 率 、 空 间 效率 和 代码 复杂 度 之 
间 如 何 取 舍 。 


1. 简单 解法 

这 个 解法 简单 且 高 效 (时 间 上 ),， 使 用 一 个 整数 到 整数 的 散 列 映射 。 这 个 算法 会 迭代 整个 数 
组 ， 对 于 元 素 x， 在 散 列 表 中 查找 sum -x， 知 存在 就 打印 (x, sum 一 x)。 将 x 加 入 散 列 表 ， 然 后 继续 
处 理 下 一 个 元 素 。 


2. 另 一 种 解法 

首先 ， 让 我 们 从 定义 人 手 。 试 着 要 找 一 对 总 和 为 z 的 数 ， 则 x 的 补 数 为 ?=-x (也 即 ， 与 zx 相 加 得 
z 的 数 )。 举 个 例子 ， 若 要 找 一 对 总 和 为 12 的 数 ， 那 么 ，-$ 的 补 数 为 17。 

现在 ， 假 设 有 这 个 已 排 好 序 的 数组 : {-2 -1 8 3 5 6 7 9 13 14}。 邻 first 指 向 数组 开头 ， 
last 指 向 数组 结尾 。 要 找 出 first 的 补 数 ， 就 将 last 往 回 移 动 , 直至 找到 补 数 。 如 果 first + last 
< sum, 则 数组 中 不 存在 first 的 补 数 , 因此 可 以 向 前 移动 first, 等 到 first 比 last 大 时 停止 操作 。 

为 什么 这 么 做 就 能 找 出 first 的 所 有 补 数 ? 因为 这 个 数组 是 排 好 序 的 ， 而 且 我 们 是 从 最 小 的 
数字 开始 逐一 尝试 的 。 当 first 与 1ast 的 总 和 小 于 sum 时 ,， 可 以 确定 ,就算 继续 尝试 更 小 的 数 ( 像 
last 那 样 往 回 移动 ) 也 找 不 到 补 数 。 




















图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 





324 第 9 章 解 题 技巧 





为 什么 这 么 做 可 以 找 出 last 的 所 有 补 数 ?” 因为 所 有 数值 对 必定 由 first 和 1ast 组 成 。 找 出 
first 的 所 有 补 数 ， 就 等 于 找 出 了 1last 的 所 有 补 数 。 


1 void printpairSums(int[] array, int sum) { 

2 Arrays.sort(array); 

3 int first = 0) 

4 int last = array.length - 1; 

5 while (first < last) { 

6 int s = array[first] + array[last]; 

了 if (s == Sum) { 

8 System.out.println(array[first] + “ “ + array[last]); 


9 first++; 

16 last--; 

11 } else { 

12 if (s < sum) first++; 
13 else last--; 

14 } 

15 

16 } 


17.13 有 个 简单 的 类 似 结 点 的 数据 结构 BiNode ， 包 含 两 个 指向 其 他 结 点 的 指针 。 数 据 结 
构 BiNode 可 用 来 表示 二 叉 树 ( 其 中 node1 为 左 子 结 点 ,node2 为 右 子 结 点 ) 或 双向 链表 ( 其 中 node1 
为 前 趋 结 点 ，node2 为 后 继 结 点 )。 编 写 一 个 方法 ， 将 二 叉 查 找 树 (用 BiNode 实 现 ) 转换 为 双向 
链表 。 要 求 所 有 数值 的 排序 不 变 , 转换 操作 不 得 引入 其 他 数据 结构 ( 即 直接 操作 原先 的 数据 结构 )。 
(第 105 页) 


解法 

此 题 看 似 复杂 , 不 过 , 运用 递归 就 能 实现 得 相当 优美 。 要 解决 此 题 ， 你 需要 对 递归 有 非常 深 
刻 的 理解 。 

设想 有 一 棵 简单 的 二 又 查找 树 : 








convert 方 法 应 该 将 它 转 换 成 下 面 的 双向 链表 : 
0<x<->1<->2<->3<x<->4<->5<->6 
下 面 我 们 会 从 根 结 点 开始 〈 结 点 4 )， 以 递归 方式 解决 问题 。 
我 们 知道 , 树 的 左右 两 半 会 在 双向 链表 里 形成 它们 自己 的 子 部 分 (也 即 , 它们 在 链表 里 的 位 
置 是 连续 的 )。 那 么 ， 若 能 以 递归 方式 将 左 子 树 和 右 子 树 转 换 成 双向 链表 ， 我 们 有 办 法 从 这 些 子 
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部 分 构建 出 最 终 的 链表 吗 ? 
当然 有 ! 直接 合并 这 两 个 子 部 分 即 可 。 
相关 伪 码 大 致 如 下 : 





1 BiNode convert(BiNode node) { 

2 BiNode left = convert(node.1left); 

3 BiNode right = convert(node.right); 
4 mergeLists(left, node, right); 

5 return left; // 左边 的 开头 

6 } 


为 了 实现 上 述 伪 码 的 琐碎 细节 ， 我 们 需要 取得 每 个 链表 的 表 头 和 表 尾 ， 有 以 下 几 种 做 法 。 


解法 1 : 附加 数据 结构 
第 一 种 ， 也 是 比较 简单 的 一 种 方法 ， 是 创建 一 个 NodePair 数 据 结 构 ， 只 包含 链表 的 表 头 和 
表 尾 。 然 后 ，convert 方 法 就 可 以 返回 一 个 NodePair 对 象 。 
下 面 是 这 种 做 法 的 实现 代码 。 








1 private class Nodepair { 

2 BiNode head; 

3 BiNode tail; 

4 

5 public Nodepair(BiNode head, BiNode tail) { 
6 this.head = head; 

了 this.tail = tail; 

8 } 

9 +} 


11 public Nodepair convert(BiNode root) { 
12 if (root == null) { 

13 return null; 

14 } 


16 NodePair part1 = convert(root.nodel); 
17 NodePair part2 = convert(root.node?2); 


19 if (part1 != null) { 
26 concat(part1.tail, root); 
21 } 


23 if (part2 != null) { 
24 concat(root, part2.head); 
25 } 


27 return new NodePair(part1 == null ? root : part1.head, 
28 part2 == null ? root : part2.tail); 
29 } 


31 public static void concat(BiNode x, BiNode y) { 
32 x.node2 = y; 

33 y.nodel = x; 

34 } 
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上 面 的 代码 仍 是 在 BiNode 数 据 结构 里 进行 转换 操作 ， 我 们 只 是 借用 NodePair 来 返回 额外 的 
数据 。 另 一 种 方式 是 使 用 有 两 个 BiNode 的 数组 , 可 实现 相同 的 目的 , 但 这 么 做 会 显得 有 些 凌 乱 ( 没 
错 ， 面 试 官 喜欢 干净 的 代码 ， 特 别 是 在 面试 中 )。 

做 得 不 错 ， 不 过 ， 要 是 不 必用 到 额外 的 数据 结构 ， 岂 不 更 好 ? 是 的 ,我们 可 以 。 


解法 2: 取 回 表 尾 
之 前 用 NodePair 返 回 链表 的 表 头 和 表 尾 ， 现 在 改 为 只 返回 表 头 ， 然 后 借助 表 头 找到 链表 的 
表 尾 。 























1 public static BiNode convert(BiNode root) { 
2 if (root == null) { 

3 return null; 

4 } 

5 

6 BiNode part1 = convert(root.nodel1); 
7 BiNode part2 = convert(root.node2); 
8 

9 if (part1 != nul1) { 

16 concat(getTail(part1), root); 

11 } 

于 之 

13 if (part2 != null) { 

14 concat(root, part2); 

15 } 

16 

17 return part1 == null ? root : part1; 
18 } 

19 


26 public static BiNode getTail(BiNode node) { 
2 if (node == null) return null; 
22 while (node.node2 != null) { 


23 node = node.node2; 

24 } 

25 return node; 

26 } 

除了 调用 getTail， 这 上段 代码 与 解法 1 几乎 完全 相同 ,但 这 么 做 效率 并 不 是 很 高 。 深 度 为 4 的 
叶 结 点 会 被 getTail 方 法 访问 4 次 (在 该 叶 结 点 之 上 有 几 个 结 点 就 访问 几 次 )， 导 致 整体 运行 时 间 


为 OW”)， 其 中 为 树 的 结 点 数 。 


解法 3: 构造 一 个 环 状 链 表 

在 解法 2 的 基础 上 ， 可 以 构建 第 三 种 也 是 最 后 一 种 解法 。 

这 种 做 法 需要 用 BiNode 返 回 链表 的 表 头 和 表 尾 。 具 体 是 将 每 个 链表 当 作 一 个 环形 链表 的 表 头 
返回 ， 然 后 ， 直 接 调 用 head.node1 就 能 取得 表 尾 。 





























1 public static BiNode convertToCircular(BiNode root) { 
2 if (root == null) { 

3 return null; 
4 


} 
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5 

6 BiNode part1 = convertToCirc 
7 BiNode part3 = convertToCirc 
8 

9 if (part1 == null && part3 = 
16 root .node1 = root; 

TL root.node2 = root; 

12 return root; 

13 } 

14 BiNode tail3 = (part3 == nul 
15 


16 /* 将 左边 加 至 根 */ 
17 if (part1 == null) { 


18 concat(part3.node1, root); 
19 } else { 

20 concat(part1.node1, root); 
21 } 

22 


23 /* 将 右边 加 至 根 */ 
24 if (part3 == null) { 


25 concat(root, part1); 
26 } else { 

27 concat(root, part3); 
28 } 

29 


36 /* 将 右边 加 至 左边 */ 

31 if (part1 != null && part3 ! 
32 concat(tail3, part1); 

33 } 


35 return part1 == null ? root : 


36 } 


38 /* 将 链表 转换 为 环形 链表 ， 然 后 断 开 
39 * 环形 连接 */ 


ular(root.nodel); 
ular(root.node2); 


= null) { 


1) ? null : part3.nodel; 


= null) { 


part1; 


46 public static BiNode convert(BiNode root) { 
41 BiNode head = convertToCircular(root); 


42 head.node1l.node2 = null; 
43 head.node1 = null; 

44 return head; 

45 } 





注意 ， 我 们 已 将 代码 主体 部 分 移 至 convertToCircular，convert 方 法 会 调用 这 个 方法 取得 


环形 链表 的 表 头 ， 然 后 断 开 环 状 连接 。 
这 种 做 法 需要 用 时 O(N)， 因 为 每 个 





结 点 平均 只 会 访问 一 次 (或 者 , 更 准确 地 说 ,是 O(1) 次 )。 


17.14 ” 哦 ， 不! 你 刚刚 写 好 一 篇 长 文 ， 却 倒霉 地 误 用 了 “查找 /替换 ”， 不 愤 删 除了 文档 中 


所 有 空格 、 标 点 ， 大 写 变 成 小 写 。 比 如 


， 句 子 “| reset the computer. lt still didn't boot!”( 我 重 


启 了 电脑 ， 但 还 没 启动 好 ! ) 变 成 了 “iresetthecomputeritstilldidntboot”。 你 发 现 ， 只 要 能 正确 分 


离 各 个 单词 ,加 标点 和 调整 大 小 写 都 不 成 问题 。 大 部 分 单词 在 字典 里 都 找 得 到 ， 有 些 字符 串 如 专 


有 名 词 则 找 不 到 。 
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给 定 一 个 字典 (一 组 单词 )， 设 计 一 个 算法 ， 找 出 拆 分 一 连 串 单词 的 最 佳 方式 。 这 里 “最 佳 ” 
的 定义 是 ， 解 析 后 无 法 辨识 的 字符 序列 越 少 越 好 。 

举 个 例子 , 字符 串 “jesslookedjustliketimherbrother” 的 最 佳 解 析 结 果 为 “JESS looked just 
like TIM her brother”， 总 共有 7 个 字符 无 法 辨别 ， 全 部 显示 为 大 写 ， 以 示 区 别 。( 第 105 页 ) 


解法 

有 些 面试 官 喜欢 开门 见 山 ,给 你 具体 的 问题 ,也 有 的 面试 官 喜欢 告诉 你 一 堆 不 必要 的 上 下 文 ， 
就 像 此 题 一 样 。 遇 到 这 种 情况 ， 最 好 将 问题 好 好 梳理 一 下 ， 找 出 到 底 要 做 什么 。 

此 题 的 关键 是 要 找到 一 种 方法 ,将 字符 串 拆 分 为 几 个 单词 ,使 得 解析 后 剩 下 的 字符 越 少 越 好 。 

注意 ,我 们 并 不 打算 试图 去 “理解 ”字符 串 ,“thisisawesome” 可 以 解析 为 “this is awesome”， 
同样 可 以 解析 为 “this is a we some”。 

此 题 的 重点 在 于 找到 一 种 方法 ， 从 子 问题 的 角度 来 定义 问题 解法 ( 也 即 解 析 后 的 字符 串 )。 
一 种 做 法 是 递归 访问 整个 字符 串 。 在 每 个 时 间 点 上 ， 最 佳 解析 是 从 两 种 可 能 决定 中 择优 而 取 。 

(1) 在 这 个 字符 后 插入 一 个 空格 。 

(2) 不 在 这 个 字符 后 插入 一 个 空格 。 

我 们 将 以 字符 串 thit 为 例 过 一 遍 此 题 的 解法 ， 如 下 所 示 。 为 了 清楚 起 见 ， 我 们 将 使 用 以 下 
记号 : 
口 无 效 单词 (字典 里 找 不 到 的 ) 全 部 大 写 ; 
口 有 效 的 单词 加 下 划 线 ; 
口 结合 在 一 起 的 字符 ( 字符 之 间 没 有 空格 ) 加 粗 表 示 。 

这 些 在 字符 串 里 的 加 粗 字 符 仍 处 于 “ 待 解析 ”的 状态 , 我 们 还 未 决定 这 些 字符 是 有 效 的 还 是 
无 效 的 〈 在 字典 中 找 不 找 得 到 )。 









































1 p(thit) 

= min(T + p(hit), p(thit)) --> 1 inv. 

3 T+ p(hit) = min(T + H + p(it), T + p(hit)) --> 1 inv. 

4 T+H+ Pp(it) = min(T + H+i+ p(t), T+H+ p(it)) --> 2 
5 T+H+i+p(t)=T+H+i+T= 3 invalid 

6 T+H+ Pp(it)=T+H+ it = 2 invalid 

7 T+ p(hit) = min(T + hi + p(t), T + p(hit)) --> 1 inv. 

8 T+hi+ p(t)=T+hi+T= 2 invalid 





9 T+ pl(hit) = T + hit = 1 invalid 

16 p(thit) = min(TH + p(it), p(thit)) --> 2 inv. 

1 TH + p(it) = min(TH + i + p(t), TH + p(it)) --> 2 inv. 
12 TH+i+ p(t)= TH+i+T= 3 invalid 

13 TH + p(it) = TH + it = 2 invalid 

14 p(thit) = min(THI + p(t), p(thit)) --> 4 inv. 

15 THI + p(t) = THI + T = 4 invalid 

16 p(thit) = THIT = 4 invalid 








在 上 述 步骤 中 ， 请 注意 每 一 层 都 分 为 两 部 分 。 第 一 部 分 分 割 字符 串 ， 而 第 二 部 分 则 进行 
结合 。 


例如 ， 当 首次 调用 p(thit) 时 ， 当 前 被 解析 的 字符 就 是 第 一 个 t， 会 递归 到 两 个 方向 。 第 一 
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个 (第 3 行 ) 会 在 t 后 面 插入 一 个 空白 ， 然 后 试 着 找 出 解析 hit 的 最 佳 方 式 。 第 二 个 (第 10 行 ) 会 
试 着 找 出 t 和 h 之 间 没 有 空白 的 最 佳 解析 方式 。 重复 执行 上 述 动作 , 最 终 就 会 得 到 字符 串 所 有 可 能 
的 解析 方式 。 

下 面 是 该 解法 的 实现 代码 。 为 了 简单 起 见 ， 我 们 实现 该 算法 时 上 只 返回 无 效 字 符 的 个 数 。 








1 public int parseSimple(int wordStart, int wordEnd) { 

2 if (wordEnd >= sentence.length()) { 

3 return wordEnd - wordStart; 

4 } 

5 

6 String word = sentence.substring(wordStart, wordEnd + 1); 
7 

8 /* 切断 当前 的 单词 */ 

9 int bestExact = parseSimple(wordEnd + 1, wordEnd + 1); 
16 if (!dictionary.contains(word)) { 

11 bestExact += word.length(); 

12 } 

13 


14 /* 扩展 当前 的 单词 */ 
15 int bestExtend = parseSsimple(wordStart, wordEnd + 1); 


17 /* 找 出 最 佳 单词 */ 
18 return Math.min(bestExact, bestExtend); 


这 段 代码 还 可 以 进行 两 处 大 的 优化 。 

口 有 些 递归 重复 了 。 例 如 ， 在 前 面 的 解析 示例 中 ， 我 们 重复 计算 了 it 的 最 佳 解析 方式 。 其 
实 第 一 次 算出 来 后 就 可 以 将 结果 缓存 起 来 ， 以 供 之 后 使 用 。 运 用 动态 规划 法 就 可 以 实现 
这 一 点 。 

口 在 某 些 情况 下 ， 我 们 或 许可 以 预测 出 某 一 个 解析 将 产生 无 效 字符 串 。 例 如 ， 假 设 正 试 着 
解析 字符 串 xten， 但 并 不 存在 以 xt 开 头 的 单词 。 然 而 ， 目 前 的 解法 还 是 会 尝试 解析 字符 
串 为 xt + p(en)、xte + p(n) 和 xten。 每 一 次 都 会 发 现 这 样 的 单词 在 字典 里 并 不 存在 。 
相反 ， 我 们 应 该 在 x 后 面 加 上 空白 ， 并 从 这 里 开始 进行 最 佳 解析 。 不 过 ， 怎 样 才能 快速 判 
朵 不 存在 以 xt 开 头 的 单词 呢 ? 答案 是 使 用 trie。 

下 面 是 上 述 两 处 优化 的 实现 代码 。 














1 public int parseOptimized(int wordStart, int wordEnd， 

2 Hashtable<Integer, Integer> cache) { 
3 if (wordEnd >= sentence.length()) { 

4 return wordEnd - wordStart; 

5 

6 if (cache.containsKey(wordStart)) { 

7 return cache.get(wordStart) ; 

8 } 

9 

16 String currentWord = sentence.substring(wordStart, wordEnd + 1); 
11 


12 /* 检查 前 级 是 否 在 字典 里 (false --> 部 分 匹配 ) */ 
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13 boolean validpartial = dictionary.contains(currentWord, false); 
14 
15 /* 切断 当前 的 单词 */ 
16 int bestExact = parseOptimized(wordEnd + 1, wordEnd + 1, cache); 
17 
18 /* 若 完 整 字符 囊 不 在 字典 里 ， 算 作 无 效 单词 */ 
19 if (!validPartial || !dictionary.contains(currentWord, true)) { 
20 bestExact += currentWord.1length(); 
21 } 
22 
23 /* 扩展 当前 的 单词 */ 
24 int bestExtend = Integer.MAX VALUE; 
25 if (validPartial) { 
26 bestExtend = parseOptimized(wordStart, wordEnd + 1, cache); 
27 } 
28 
29 /* 找 出 最 佳 单词 */ 
36 int min = Math.min(bestExact, bestExtend); 
31 cache.put(wordStart,，min); // 缓存 结果 
32 return min; 
33 } 























注意 , 我 们 使 用 了 散 列 表 来 缓存 结果 ， 键 为 单词 开头 的 索引 。 也 就 是 说 ,我 们 缓存 的 是 字符 
串 剩余 部 分 的 最 佳 解析 方式 。 
我 们 可 以 调整 代码 , 返回 解析 后 的 完整 字符 串 , 但 这 样 做 稍微 有 点 复杂 。 我 们 需要 使 用 名 为 


Result 的 包 庄 类 ， 这 样 才 能 同时 返回 无 效 字 符 的 个 数 和 最 佳 


按 引用 传 值 。 
1 public class Result { 
2 public int invalid = Integer.MAX VALUE; 
3 public String parsed = “”; 
4 public Result(int inv, String p) { 
5 invalid = inv; 
6 parsed = p; 
7 } 
8 
9 public Result clone() { 
16 return new Result(this.invalid, this.parsed); 
ull } 
12 
13 public static Result min(Result ri, Result r2) { 
14 if (Pr1 == null) { 
15 return r2; 
16 } else if (r2 == null) { 
17 return ri1; 
18 } 
19 return r2.invalid < ri1.invalid ? r2 : rl; 
26 } 
21 } 
22 
23 public Result parse(int wordStart, int wordEnd， 


[= 

















和 。 要 是 以 C++ 实 现 的 话 





字符 日 


Hashtable<Integer, Result> cache) { 
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25 if (wordEnd >= sentence.length()) { 


26 return new Result(wordEnd - wordSstart， 

27 sentence.substring(wordStart).toUpperCase()); 

28 

29 if (cache.containsKey(wordStart)) { 

36 return cache.get(wordStart).clone(); 

31 } 

32 String currentWord = sentence.substring(wordStart, wordEnd + 1); 
33 boolean validpartial = dictionary.contains(currentWord, false); 
34 boolean validExact = ValidPartial && 

35 dictionary.contains(currentWord, true); 
36 

37 /* 切断 当前 的 单词 */ 

38 Result bestExact = parse(wordEnd + 1, wordEnd + 1, cache); 
39 if (validExact) { 

46 bestExact.parsed = currentWord + “ ” + bestExact.parsed; 
41 } else { 

42 bestExact.invalid += currentWord.length(); 

43 bestExact.parsed = currentWord.toUpperCase() + “ ”+ 

44 bestExact .parsed; 

45 } 

46 


47 /* 扩展 当前 的 单词 */ 

48 Result bestExtend = null; 

49 if (validPartial) { 

56 bestExtend = parse(wordStart, wordEnd + 1, cache); 
51 } 

5 

53 /* 找 出 最 佳 单词 */ 

54 Result best = Result.min(bestExact, bestExtend); 
55 cache.put(wordStart, best.clone()); 

56 return best; 

57 } 


在 动态 规划 问题 中 ,对 于 如 何 缓存 对 象 , 要 非常 小 心 。 若 缓存 的 东西 是 对 象 而 非 基本 数据 类 
型 ， 很 可 能 需要 复制 该 对 象 。 这 可 以 从 上 面 第 30 一 55 行 代码 看 出 。 如 果 不 加 复制 ， 后 续 对 parse 
的 调用 就 会 自 改 缓存 里 的 值 。 








9.18 高 难度 题 


18.1 编写 一 个 函数 ， 将 两 个 数字 相 加 。 不 得 使 用 + 或 其 他 算术 运算 符 。( 第 105 页 ) 


解法 

遇 到 这 类 问题 , 第 一 反应 是 我 们 需要 跟 比特 位 打交道 , 八 九 不 离 十 。 何 出 此 言 ?” 原因 很 简单 ， 
连 加 号 (+) 都 不 能 用 了 ， 还 有 其 他 选择 吗 ? 再 说 了 ， 计 算 机 在 计算 时 就 是 跟 比特 位 打交道 的 。 

接 下 来 ,我 们 应 该 着 眼 于 切实 理解 加 法 是 怎么 工作 的 。 我 们 可 以 过 一 遍 加 法 问题 ， 看 看 自己 
能 否 悟 出 新 东西 一 一 某 种 模式 ， 然 后 ， 试 试 能 否 用 代码 来 实现 。 

闲话 少 说 ， 下 面 就 来 探讨 一 个 加 法 问题 ， 并 以 十 进 制 运算 ， 这样 更 容易 理解 。 
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要 做 759 + 674 加 法 运算 ， 通 常会 将 每 个 数字 的 个 位 数 ( digit[8] ) 相 加 、 进 位 ， 然 后 将 每 
个 数字 的 十 位 数 (digit[1] ) 相 加 、 进 位 ， 依 此 类 推 。 二 进 制 加 法 也 可 以 采取 同样 的 做 法 : 各 
位 数 相 加 ， 必 要 时 进位 。 

有 没有 办 法 让 程序 简单 一 点 呢 ? 当然 有 ! 设想 一 下 ,把 “ 相 加 ”和 “进位 ”等 步骤 分 开 ， 也 
就 是 说 ， 像 下 面 这 么 做 。 

() 将 739 和 674 相 加 ， 但 “ 忘 了 ”进位 ， 得 到 323。 

(2) 将 739 和 674 相 加 ， 但 只 进位 ， 不 会 将 各 位 数 加 在 一 起 ， 得 到 1110。 

(3) 将 前 面 两 步 操作 的 结果 加 起 来 一 一 递归 执行 步 又 (0) 和 步骤 (2) 描 述 的 过 程 : 1110 + 323 = 
1433。 

那么 ， 对 于 二 进 制 ， 该 怎么 做 ? 

(1) 若 将 两 个 二 进 制 数 加 在 一 起 ， 但 忘记 进位 ， 只 要 a 和 2 的 ;位 相同 ( 丝 为 0 或 丝 为 1 )， 总 和 
的 i 位 就 为 0。 这 实质 上 就 是 异 或 操作 (XOR )。 

(2) 若 将 两 个 数字 加 在 一 起 ,但 只 进位 ， 只 要 a 和 2 的 六 1 位 尼 为 1， 总 和 的 ;位 就 为 1。 这 实质 上 
就 是 位 与 (AND ) 加 上 移 位 操作 。 

(G3) 接着， 递归 执行 步骤 (1) 和 (2)， 直 至 没有 进位 为 止 。 

下 面 是 该 算法 的 实现 代码 。 












































1 public static int add(int a, int b) { 

2 if (b == 60) return a; 

3 int sum = a ^ b;j // 相 加 但 不 进位 

4 int carry = (a & b) << 1; // 进位 ， 但 不 相 加 
5 return add(sum，carry); // 递归 

6 


} 
要 求 我 们 实现 基本 算术 运算 ， 比 如 加 法 和 减法 ,这 类 问题 比较 常见 。 这 些 问 题 的 关键 在 于 深 
和 挖掘 这 些 运 算 通 常 是 怎么 实现 的 ， 这 样 就 可 根据 给 定 问题 的 限制 重新 实现 相关 运算 。 


18.2 ”编写 一 个 方法 ， 洗 一 副 牌 。 要 求 做 到 完美 洗 牌 换言之 ， 这 副 牌 52! 种 排列 组 合 出 现 
的 概率 相同 。 假 设 给 定 一 个 完美 的 随机 数 发 生 器 。( 第 106 页 ) 

解法 

这 个 面试 题 非常 有 名 ,算法 也 很 知名 。 掌 握 这 个 算法 的 人 却 不 多 ， 如 果 你 还 不 是 其 中 之 一 ， 
还 请 继续 往 下 看 。 

假定 有 个 数组 ， 含 "个 元 素 ， 类 似 如 下 : 

[ [2] [3] [4] [5] 

利用 简单 构造 法 ， 我 们 不妨 先 问 问 自己 : 假定 有 个 方法 shuffle(... ) 对 n - 1 个 元 素 有 效 ， 
我 们 可 以 用 它 来 打 乱 个 元 素 的 次 序 吗 ? 

当然 可 以 ,而 且 非 常 容易 实现 。 我 们 会 先 打 乱 前 n -1 个 元 素 的 次 序 ， 然 后， 取出 第 个 元 素 ， 
将 它 与 数组 中 的 元 素 随 机 交换 。 就 这 么 简单 ! 

递归 解法 的 算法 类 似 如 下 : 
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/* lower 和 higher ( 含 ) 之 间 的 随机 数 */ 
int rand(int lower, int higher) { 
return lower + (int)(Math.random() * (higher - lower + 1)); 


} 


int[] shuffleArrayRecursively(int[] cards, int i) { 
if (i == 6) return cards; 


ovOmFwN 哺 


9 shuffleArrayRecursively(cards, i - 1); // 打 乱 先前 部 分 的 次 数 
186 int k = rand(8，i); // 随机 挑选 索引 进行 交换 


12  /* 交换 元 素 k 和 i */ 

13 int temp = cards[k]; 
14 cards[k] = cards[i]; 
15 cards[i] = temp; 


17  /* 返回 元 素 次 序 被 打 乱 的 数组 */ 
18 return cards; 

















以 迭代 方式 实现 的 话 , 这 个 算法 又 会 是 什么 样 ? 让 我 们 先 考 虑 一 下 。 我 们 要 做 的 就 是 遍历 整 


个 数组 ， 对 每 个 元 素 i， 将 array[i] 与 0 和 i ( 含 ) 之 间 的 随机 元 素 交 换 。 
其 实 ， 这 个 算法 一 点 也 不 绕 ， 很 适合 以 迭代 方式 实现 : 


1 void shuffleArrayInteratively(int[] cards) { 





2 for (int i = 6;j i «< cards.length; i++) { 
3 int k = rand(60, i); 

4 int temp = cards[k]; 

5 cards[k] = cards[i]; 

6 cards[i] = temp; 

7 } 

8 } 


这 样 就 可 以 用 迭代 法 实现 该 算法 了 。 


18.3 ”编写 一 个 方法 ， 从 大 小 为 n 的 数组 中 随机 选 出 m 个 整数 。 要 求 每 个 元 素 被 选中 的 概率 


相同 。( 第 106 页 ) 


解法 
与 18.2 类 似 ， 我 们 可 以 利用 简单 构造 法 ， 以 递归 方式 处 理 此 题 。 


假定 有 个 算法 可 以 从 包含 n -1 个 元 素 的 数组 中 随机 抽出 m 个 元 素 ， 我们 可 以 使 用 该 算法 从 包 


含 n 个 元 素 的 数组 中 随机 抽出 m 个 元 素 吗 ? 
我 们 可 以 先 从 前 n - 1 个 元 素 中 随机 抽出 mm 个 元 素 。 然 后 ， 只 需 决 定 array[n] 是 否 应 该 捐 
subset (从 中 随机 抽出 一 个 元 素 )。 一 种 简单 的 做 法 是 从 0 到 z 中 随机 挑选 一 个 数 F。 若 上 < m， 








入 
则 


将 array[n] 插 入 subset[k]。 将 array[n] 插 入 subset ( 按 比 例 概 率 ) 以 及 从 subset 中 随机 移 除 





一 个 元 素 ， 两 者 都 很 “公平 ”。 
这 个 递归 算法 的 伪 码 大 致 如 下 : 


1 int[] pickMRecursively(int[] original, int m, int i) { 
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2 if (i + 1 == m) { // 终止 条 件 

3 /* 返回 original 数 组 的 前 m 个 元 素 */ 

4 } else if (+m>m) 

5 int[] subset = pickMRecursively(original, m, i - 1); 
6 int k = random value between 6 and i, inclusive 
7 if (Kk < m) { 

8 subset[k] = original[i]; 

9 

16 return subset; 

站 和 

Peturn null; 

13 } 


这 个 算法 的 迭代 实现 写 起 来 更 明晰 。 在 这 种 做 法 中 , 我 们 会 先 创建 数组 subset, 并 将 它 初 始 
化 为 original 数 组 的 前 m 个 元 素 。 然 后 ， 从 元 素 m 开 始 ， 壕 代 访 问 original 数 组 ， 只 要 k<m， 就 
将 array[i] 插 入 subset 数 组 的 ( 随机 选 出 的 ) 位 置 k。 





int[] pickMIteratively(int[] original, int m) { 
int[] subset = new int[m]; 


for (int i = 6; i < m ; i++) { 
subset[i] = original[i]; 


} 


9 /* 访问 original 数 组 的 剩余 元 素 */ 
10 for (int i = mi i < original.length; i++) { 


1 
2 
六 
4 /* 用 original 数 组 的 前 m 个 元 素 填 入 Subset */ 
5 
6 
7 
8 


1 int k = rand(8，i); // 取得 6 到 i ( 含 ) 之 间 的 随机 数 
12 if (k < m) { 

13 subset[k] = original[i]; 

14 } 

15 } 

16 

17 return subset; 

18 } 


一 点 也 不 奇怪 ， 这 两 个 解法 与 打 乱 数组 的 算法 非常 相似 。 

18.4 ”编写 一 个 方法 ， 数 出 0 到 n( 舍 ) 中 数字 2 出 现 了 几 次 。( 第 106 页 ) 

解法 

面 对 此 题 ， 我 们 想到 的 第 一 种 做 法 会 是 ， 也 应 该 是 蛮 力 法 。 记 住 ， 面 试 官 希 望 看 到 你 是 怎么 
解 题 的 ， 移 给 出 查 力 解法 也 是 非常 不 错 的 开始 。 

1 /* 数 一 数 9 到 n 中 数字 2 出 现 的 次 数 */ 


2 int numberOf2sInRange(int n) { 
3 int count = 0; 











4 for (int i = 2; i <= nj i++) { // 不 妨 直接 从 2 开始 
5 count += numberOf2s(i); 

6 } 

7 return count; 

8 } 


图 灵 社 区 会 员 cindy282694 专 享 尊重 版 权 


9.18 高 难度 题 335 





9 

16 /* 数 出 某 个 数字 中 有 几 个 2 */ 
11 int numberOf2s(int n) { 
12 int count = 0; 

13 while (n > 6) { 


14 if (n % 160 == 2) { 
工 5 count++; 

16 } 

17 n=n/ 10; 

18 } 

19 return count; 

26 } 





其 中 有 个 地 方 应 该 注意 ， 就 是 最 好 将 numberof2s 独 立 写成 一 个 方法 ， 这 样 一 来 ， 代 码 也 许 
更 加 清晰 ， 也 会 展现 出 你 注重 代码 的 干净 齐整 。 


改进 后 的 解法 
之 前 的 解法 是 从 一 个 范围 内 的 数字 来 看 ,现在 我 们 从 数字 的 每 个 位 来 观察 问题 。 假 设 有 下 面 
一 个 数字 序列 : 
9 1 2 3 4 5 6 7 8 9 


16 11 12 13 14 15 16 17 18 19 
26 21 22 23 24 25 26 27 28 29 








119 111 112 113 114 115 116 117 118 119 
由 观察 可 知 ， 每 10 个 数字 中 , 最 后 一 位 为 2 的 情况 大 概 会 出 现 一 次 ， 因 为 2 在 连续 10 个 数 中 都 
会 出 现 一 次 。 实 际 上 ， 任 意 位 为 2 的 概率 大 概 是 1/10。 
之 所 以 说 “大 概 ”， 是 因为 存在 边界 条 件 ( 非常 常见 )。 例 如， 在 1 到 100 之 间 ， 十 位 数 为 2 的 
概率 正好 为 10。 然 而 ， 在 1 到 37 之 间 ， 十 位 数 为 2 的 概率 就 会 比 110 还 大 。 
下 面 逐一 分 析 digit < 2、digit = 2 和 digit > 2 三 种 情况 ， 就 能 算出 准确 的 比率 。 
@ 情况 1: digit<2 
以 x = 61 523 和 d = 3 为 例 ， 可 以 看 出 x[d] = 1 (也 即 x 的 第 d 位 数 为 1 )。 第 3 位 数 为 2 的 范围 
是 2668 - 2999、12 86686 - 12 999、22 668 - 22 999、32 668 - 32 999、42 8660 - 42 999 
和 52 668 - 52 999， 还 没 到 范围 62 66@ - 62 999， 因 此 第 3 位 数 总 共有 6000 个 2。 这 个 数量 等 
于 范围 1 到 60 000 里 第 3 位 数 为 2 的 数量 。 
换 句 话说 ,我 们 可 以 将 原来 的 数 往 下 降 至 最 近 的 10， 然 后 再 除 以 10， 就 可 以 算出 第 d 位 数 
为 2 的 数量 。 
if x[d] <2: count2sInRangeAtDigit(x, d) = 
d+1 





























let y =round down to nearest 10 

returny/10 

e@ 情况 2: digit> 2 
现在 ,我 们 再 来 看 看 x 的 第 d 位 数 大 于 2( x[d] > 2 ) 的 情况 。 基 本 上 ， 我们 可 以 运用 之 前 相 
同 的 逻辑 ， 确 认 范 围 。- 63 525 里 第 3 位 数 为 2 的 数量 与 范围 。- 78@ 886 是 相同 的 。 因 此 ， 之 前 
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是 往 下 降 ， 现 在 是 往 上 升 。 
if x[d] > 2: count2sInRangeAtDigit(x, d) = 
let y = round up to nearest 10°! 
returny/10 
e@ 情况 3: digit=2 
最 后 这 种 情况 可 能 是 最 棘手 的 ， 不 过 仍 可 套用 之 前 的 逻辑 。 以 x = 62 523 和 d = 3 为 例 ， 由 
之 前 的 逻辑 可 得 到 相同 的 范围 (也 即 范围 2668 - 2999, 12 666 - 12 999, ..., 52 866 - 52 999 )。 
在 最 后 余下 的 62 668 - 62 523 这 个 局 部 范围 里 ， 第 3 位 数 为 2 的 数量 有 多 少 ? 其 实 ， 再 显而易见 
不 过 了 。 只 有 524 个 (62 666, 62 861, .…, 62 523 )。 
if x[d] > 2: count2sInRangeAtDigit(x, d) = 
let y = round down to nearest 10¢"! 
let z = right side of x (i.e., x % 109) 
returny/10+z+1 
现在 ， 只 需 迭 代 访 问 数字 中 的 每 个 位 数 。 相 关 代码 实现 起 来 相当 直接 。 








1 public static int count2sInRangeAtDigit(int number, int d) { 
2 int powerOf16 = (int) Math.pow(106, d); 

3 int nextPowerOf16 = powerOf16 * 10; 

4 int right = number % powerOf16; 

5 

6 int roundDown = number - number % nextPowerOf10; 
7 int roundUp = roundDown + nextPowerOf10; 

8 

9 int digit = (number / powerOf16) % 16; 

16 if (digit < 2) { // 若 第 digit 位 数 …… 

11 return roundDown / 108; 

12 } else if (digit == 2) { 

13 return roundDown / 16 + right + 1; 

14 } else { 

15 return roundUp / 16; 

16 } 

17 } 

18 


19 public static int count2sInRange(int number) { 
26 int count = 6 

21 int len = String.valueOof(number).1length(); 
22 for (int digit = 6; digit < len; digit++) { 


23 count += count2sInRangeAtDigit(number, digit); 
24 } 

25 return count; 

26 } 


解决 此 题 时 ， 需 要 进行 非常 仔细 的 测试 ， 务必 列 全 一 系列 的 测试 用 例 ， 然 后 逐一 测试 
验证 。 
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18.5 有 个 内 含 单词 的 超大 文本 文件 ， 给 定 任意 两 个 单词 ， 找 出 在 这 个 文件 中 这 两 个 单 
词 的 最 短 距离 (也 即 相隔 几 个 单词 )。 有 办 法 在 O(1) 时 间 里 完成 搜索 操作 吗 9 解法 的 空间 复杂 度 
如 何 ? (第 106 页 ) 

解法 

在 此 题 中 ， 我 们 假设 单词 word1 和 word2 谁 在 前 谁 在 后 无 关 紧 要 ， 当 然 最 好 与 面试 官 确认 能 
否 做 此 假设 。 若 考虑 单词 前 后 顺序 的 话 ， 那 么 ， 下 面 给 出 的 代码 需要 稍 作 调 整 。 

要 解决 此 题 , 我 们 只 需 遍 历 一 次 这 个 文件 。 在 遍历 期 间 , 我 们 会 记 下 最 后 看 见 word1 和 word2 
的 地 方 ， 并 把 它们 的 位 置 存 入 lastPosWord1 和 1astPosWord2 中 。 碰 到 word1l 时 ， 就 拿 它 跟 
lastPosWword2 比 较 ， 如 有 必要 则 更 新 min， 然 后 更 新 lastPosWord1。 而 每 当 碰 到 word2 时 ， 我 们 
也 会 执行 同样 的 操作 。 遍 历 结束 后 ， 就 可 得 到 最 短 距离 。 

下 面 是 该 算法 的 实现 代码 。 






































1 public int shortest(String[] words, String word1l, String word2) { 
2 int min = Integer.MAX_ VALUE; 

3 int lastPosWord1 = -1; 

4 int lastPosWord2 = -1; 
5 for (int i = 8; i < words.length; i++) { 
6 String currentWord = words[i]; 

7 if (currentWord.equals(word1)) { 

8 lastPosWord1 = i; 


9 // 车 要 区 别 单词 的 前 后 顺序 ， 注 掉 下 面 3 行 

16 int distance = lastPosWord1 - lastPosWord2; 
11 if (lastPosWord2 >= 6 && min > distance) { 
12 min = distance; 

13 } 

14 } else if (currentWord.equals(word2)) { 

15 lastPosWord2 = i; 

16 int distance = lastPosWord2 - lastPosWord1; 
17 if (lastPosWord1 >= 6 && min > distance) { 
18 min = distance; 

19 } 

26 } 

21 } 

22 return min; 

23°} 











如 果 上 述 代 码 要 被 重复 调用 (查询 其 他 单词 对 的 最 短 距离 )， 可 以 构造 一 个 散 列 表 ， 记 录 每 
个 单词 及 其 出 现 的 位 置 。 然 后 ， 我 们 只 需 找 出 1istA 和 1istB 中 (算术 ) 差 值 最 小 的 那 两 个 值 。 
计算 1istA 和 1istB 中 元 素 最 小 差 值 有 好 几 种 方法 ， 以 下 面 的 列表 为 例 : 


listA: {1, 2, 9, 15, 25} 
listB: {4, 106, 19} 


将 这 两 个 列表 合并 为 一 个 列表 并 排序 ,在 每 个 数字 后 面 打 上 标记 , 标明 取 自 哪个 列表 。 打 标 
记 时 可 将 每 个 值 封装 在 一 个 类 里 ， 这 个 类 有 两 个 成 员 变 量 : data ( 储存 实际 值 ) 和 1istNumber。 


list: {1a, 2a, 4b, 9a, 16b, 15a, 19b, 25a} 
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现在 , 想 要 找 


出 最 短 距 离 的 话 ， 只 需 遍 历 合并 后 的 列表 ,查找 两 个 取 自 不 同 列表 的 连续 数字 





且 它 们 之 间 的 差 为 





最 小 值 。 在 上 面 的 例子 中 ， 答 案 是 最 短 距 离 为 1 (在 94a 和 10b 之 间 )。 


18.6 ”设计 一 个 算法 ， 给 定 10 亿 个 数字 ， 找 出 最 小 的 100 万 个 数字 。 假 定 计算 机 内 存 足 以 
容纳 全 部 10 亿 个 数字 。( 第 106 页 ) 


解法 


此 题 有 很 多 种 解法 ， 下 面 将 介绍 其 中 三 种 : 排序 、 小 项 堆 和 选择 排序 ( selection rank )。 


解法 1: 排序 


按 升序 排序 所 有 元 素 ， 然 后 取出 前 100 万 个 数 。 时 间 复 杂 度 为 O(n log(n))。 


解法 2: 小 项 堆 
我 们 可 以 使 用 小 项 堆 来 解 题 。 首先 , 为 前 100 万 个 数字 创建 一 个 大 项 堆 ( 最 大 元 素 位 于 堆 顶 )。 
然后 ， 遍 历 整个 数列 ， 将 每 个 元 素 搬 和 人 大 项 堆 ， 并 删除 最 大 的 元 素 。 


遍历 结束 后 ， 





我 们 将 得 到 一 个 堆 ， 刚 好 包含 最 小 的 100 万 个 数字 。 这 个 算法 的 时 间 复 杂 度 为 


O(n log(m))， 其 中 mm 为 待 查找 数值 的 数量 。 
解法 3: 选择 排序 算法 (假如 你 可 以 修改 原始 数组 ) 


在 计算 机 科学 
最 大 ) 元 素 。 





中 , 选择 排序 是 个 很 有 名 的 算法 , 可 以 在 线性 时 间 内 找到 数组 中 第 i 个 最 小 (或 








如 果 这 些 元 素 
如 下 。 








各 不 相同 , 则 可 在 预期 的 O(n) 时 间 内 找到 第 个 最 小 的 元 素 。 该 算法 的 基本 流程 


(1) 在 数组 中 随机 挑选 一 个 元 素 ， 将 它 用 作 “pivot”( 基准 )。 以 pivot 为 基准 划分 所 有 元 素 ， 
记录 pivot 左 边 的 元 素 个 数 。 


(2) 如 果 左 边关 








I 好 有 i 个 元 素 ， 则 直接 返回 左边 最 大 的 元 素 。 











(3) 如 果 左 边 元 素 个 数 大 于 i， 则 继续 在 数组 左边 部 分 重复 执行 该 算法 。 
(4) 如 果 左 边 元 素 个 数 小 于 i， 则 在 数组 右边 部 分 重复 执行 该 算法 , 但 只 查找 排 i-leftSize 的 那 











个 元 素 。 
下 面 是 该 算法 的 实现 代码 。 
1 public int partition(int[] array, int left, int right, int pivot) { 
2 while (true) { 
3 while (left <= right && array[left] <= pivot) { 
4 left++; 
5 } 
6 
7 while (left <= right && array[right] > pivot) { 
8 right--; 
9 } 
106 
11 if (left > right) { 
12 return left - 1; 
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13 } 

14 swap(array, left, right); 
15 } 

16 } 

17 


18 public int rank(int[] array, int left, int right, int rank) { 
19 int pivot = array[randomIntInRange(left, right)]; 


21 /* 分 害 ， 返 回 左 边 部 分 的 结尾 */ 
22 int leftEnd = partition(array, left, right, pivot); 


24 int leftSize = leftEnd - left + 1; 
25 if (leftSize == rank + 1) { 


26 return max(array, left, leftEnd); 

27 } else if (rank < leftSize) { 

28 return rank(array, left, leftEnd, rank); 

29 } else { 

36 return rank(array, leftEnd + 1, right, rank - leftSize); 
31 

32 } 


一 旦 找到 第 ;小 的 元 素 ， 你 就 可 以 遍历 整个 数组 ， 找 到 所 有 小 于 或 等 于 该 元 素 的 值 。 

如 果 这 些 元 素 有 重复 值 (一 般 不 大 可 能 )， 就 需要 对 这 个 算法 略 作 调整 ， 以 适应 这 一 变化 。 
不 过 ， 这 样 一 来 ， 就 不 能 保证 算法 执行 时 间 的 上 限 了 。 

有 个 算法 可 以 保证 在 线性 时 间 内 找到 第 ;小 的 元 素 ， 无 论 元 素 有 无 重复 值 。 然 而 ， 这 个 算法 
的 复杂 度 远 远 超出 了 面试 的 范围 。 若 有 兴趣 的 话 ， 请 参考 CLRS 四 人 合 著 的 《算法 导论 》 一 书 。 


18.7 给 定 一 组 单词 ， 编 写 一 个 程序 ， 找 出 其 中 的 最 长 单词 ， 且 该 单词 由 这 组 单词 中 的 其 
他 单词 组 合 而 成 。( 第 106 页 ) 


解法 

此 题 看 似 比较 复杂 ,让 我 们 先 来 简化 一 番 。 如 果 只 是 想 知道 由 列表 中 的 其 他 两 个 单词 组 成 的 
最 长 单词 ， 该 怎么 处 理 ? 

我 们 可 以 通过 遍历 整个 列表 ， 从 最 长 单词 到 最 短 单词 ， 将 每 个 单词 分 割 成 所 有 可 能 的 两 半 ， 
然后 检查 左右 两 半 是 否 在 列表 中 。 























上 述 做 法 的 伪 码 大 致 如 下 : 

1 String getLongestWord(String[] list) { 

2 String[] array = list.SortByLength(); 

3 /* 创建 map 以 便 查找 */ 

4 HashMap<String, Boolean> map = new HashMap<String, Boolean>; 
5 

6 for (String str : array) { 

7 map.put(str, true); 

8 } 

9 

16 for (String s : array) { 

11 // 切 分 成 所 有 可 能 的 两 半 

12 for (int i = 1; i < s.length(); i++) { 
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13 String left = s.substring(0, i); 
14 String right = s.substring(i); 
15 // 检查 左右 两 半 是 否 在 数组 中 
16 if (map[left] == true && map[right] == true) { 
了 到 return s; 
18 } 
19 } 
26 } 
21 return str; 
22 } 


若 知道 最 长 单词 由 另外 两 个 单词 组 合 而 成 时 ,这 么 做 非常 有 效 。 但 是 ， 若 单词 可 以 由 任意 数 
量 的 其 他 单词 组 成 ， 又 会 怎么 样 呢 ? 

在 这 种 情况 下 , 我 们 可 以 采用 非常 相似 的 做 法 ,只 修改 一 处 : 之 前 会 检查 右 半 部 分 是 否 在 数 
组 中 ， 现 在 改 为 递归 检查 右 半 部 分 可 否 由 数组 其 他 元 素 构建 出 来 。 

下 面 是 该 算法 的 实现 代码 : 


- 
3 
4 
5 
6 
7 
8 
9 


14 } 





String printLongestWord(String arr[]) { 


HashMap<String, Boolean> map = new HashMap<String, Boolean>(); 
for (String str : arr) { 
map.put(str, true); 
} 
Arrays.sort(arr，new LengthComparator()); // 按 长 度 排序 
for (String s : arr) { 
if (canBuildWord(s, true, map)) { 
System.out.println(s); 
return s; 
} 
} 


return “”; 


16 boolean canBuildWord(String str, boolean isOriginalWord, 


36 
31 } 


注意 ， 
。 这 样 一 来 ， 如 需 反 复 检 查 有 无 办 法 构造 “testingtester”， 就 只 需要 计算 一 次 。 


间 的 结 


HashMap<String, Boolean> map) { 
if (map.containsKey(str) && !isOriginalWord) { 
return map.get(str); 
} 
for (int i = 1; i «< str.length(); i++) { 
String left = str.substring(0, i); 
String right = str.substring(i); 
if (map.containsKey(left) && map.get(left) == true && 
canBuildWword(right, false, map)) { 
return true; 
} 
} 
map.put(str, false); 
return false; 








在 这 个 解法 中 , 我 们 做 了 一 个 小 小 的 优化 。 我 们 使 用 动态 规划 方法 缓存 了 多 次 调用 之 
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其 中 ,布尔 标志 isoriginalWord 用 于 完成 上 面 的 优化 。 调 用 方法 canBuildWword 时 ,会 传人 原 
始 单词 和 每 个 子 申 ， 在 算法 里 ,第 一 步 会 先 检查 缓存 里 有 无 之 前 计算 好 的 结果 。 但是, 这 里 也 有 个 
问题 : 对 于 原始 单词 ，map 会 将 这 些 单词 初始 化 为 true， 但 我 们 又 不 想 返回 true (因为 证 闻 个 能 只 
由 它 本 身 组 成 )。 因 此 ， 对 于 原始 单词 ， 我 们 会 利用 isoriginalWord 标 志 直 接 跳 过 这 项 检查 。 























18.8 给 定 一 个 字符 串 s 和 一 个 包含 较 短 字符 串 的 数组 T， 设 计 一 个 方法 ， 根 据 T 中 的 每 一 
个 较 短 字 符 串 ， 对 s 进 行 搜索 。( 第 106 页 ) 

解法 

首先 ， 创 建 s 的 后 绥 树 (suffix tree )。 举 个 例子 ， 若 单词 为 bibs， 则 这 棵 树 如 下 所 示 : 








然后 ， 只 需 在 这 棵 后 绥 树 中 搜索 查找 T 中 的 每 个 字符 串 。 注 意 ， 如 果 “B” 是 个 单词 的 话 ， 
你 会 得 到 两 个 位 置 。 





1 public class SuffixTree { 

2 SuffixTreeNode root = new SuffixTreeNode(); 
3 public SuffixTree(String s) { 

4 for (int i = 6; i < s.length(); i++) { 

5 String suffix = s.substring(i); 

6 root.insertString(suffix, i); 

学 

8 


} 

} 
9 
16 public ArrayList<Integer> search(String s) { 
11 return root.search(s); 
12 } 
13 } 
14 
15 public class SuffixTreeNode { 
16 HashMap<Character, SuffixTreeNode> children = new 
17 HashMap<Character, SuffixTreeNode>(); 
18 char value; 
19 ArrayList<Integer> indexes = new ArrayList<Integer>(); 
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26 public SuffixTreeNode() { } 


21 

2 public void insertString(String s, int index) { 
23 indexes.add(index); 

24 if (s != null && s.length() > 6) { 

25 value = s.charAt(0); 

26 SuffixTreeNode child = null; 

27 if (children.containsKey(value)) { 

28 child = children.get(value); 

29 } else { 

36 child = new SuffixTreeNode(); 

3 于 children.put(value, child); 

32 } 

33 String remainder = s.substring(1); 

34 child.insertSstring(remainder, index); 
35 } 

36 } 

37 

38 public ArrayList<Integer> search(String s) { 
39 if (s == null || s.length() == 6) { 

46 return indexes; 

41 } else { 

42 char first = s.charAt(@); 

43 if (children.containsKey(first)) { 

44 String remainder = s.substring(1); 
45 return children.get(first).search(remainder); 
46 } 

47 } 

48 return null; 

49 } 

56 } 


18.9 随机 生成 一 些 数字 并 传 入 某 个 方法 。 编 写 一 个 程序 ， 每 当 收 到 新 数字 时 ， 找 出 并 记 


录 中 位 数 。( 第 106 页) 
解法 








一 种 解法 是 使 用 两 个 优先 级 堆 ( priority heap ): 一 个 大 项 堆 ， 存 放 小 于 中 位 数 的 值 ， 以 及 一 


个 小 项 堆 , 存放 大 于 中 位 数 的 值 。 这 会 将 所 有 元 素 大 致 分 为 两 半 ， 中间 上 
堆 项 。 这 样 一 来 ， 要 找 出 中 位 数 就 是 小 事 一 桩 。 




















的 两 个 元 素 位 于 两 个 堆 的 























不 过 ,“ 大 致 分 为 两 半 ” 又 是 什么 意思 呢 ? “大 致 ”的 意思 是 ， 如 果 有 奇数 个 值 ， 其 中 一 个 


堆 就 会 多 一 个 值 。 经 观察 可 知 ， 以 下 两 点 为 真 。 








值 为 中 位 数 。 
当 要 重新 平衡 这 两 个 堆 时 ， 我 们 会 确保 maxHeap 一 定 会 多 一 个 元 素 。 
这 个 算法 说 明 如 下 。 有 新 的 值 生 成 时 ， 如 果 这 个 值 小 于 等 于 中 位 数 
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口 如 果 maxHeap.size() > minHeap.size()， 则 maxHeap.top() 为 中 位 数 。 
口 如 果 maxHeap.size() == minHeap.size()， 则 maxHeap.top() 和 minHeap.top() 的 平均 


， 则 放 入 maxHeap 中 ， 否 
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则 放 入 minHeap。 两 个 堆 的 元 素 个 数 相等 ， 或 者 maxHeap 可 能 多 一 个 元 素 。 这 个 限制 条 件 很 容易 
得 到 保证 ， 不 满足 的 话 ， 只 要 从 一 个 堆 搬 移 一 个 元 素 到 另 一 个 堆 即 可 。 通 过 查看 maxHeap 或 两 个 
堆 的 堆 顶 元 素 ， 就 能 以 常数 时 间 获 取 中 位 数 ， 而 更 新 操作 的 用 时 为 O(log(n))。 























1 private Comparator<Integer> maxHeapComparator; 
2 private Comparator<Integer> minHeapComparator; 
3 private PriorityQueue<Integer> maxHeap, minHeap; 
4 

5 public void addNewNumber(int randomNumber) { 
6 /* 注意 : addNewNumber 会 保持 下 面 的 条 件 : 

7 * maxHeap.size() >= minHeap.size() */ 

8 if (maxHeap.size() == minHeap.size()) { 

9 if ((minHeap.peek() != null) && 

16 randomNumber > minHeap.peek()) { 

11 maxHeap.offer(minHeap.poll1()); 

12 minHeap.offer(randomNumber); 

13 } else { 

14 maxHeap.offer(randomNumber); 

15 } 

16 } else { 

17 if(randomNumber < maxHeap.peek()) { 

18 minHeap.offer(maxHeap.poll1()); 

19 maxHeap.offer(randomNumber); 

26 } 

21 else { 

22 minHeap.offer(randomNumber); 

23 } 

24 } 

25 } 

26 


27 public static double getMedian() { 


28 /* maxHeap 至 少 会 跟 minHeap 一 样 大 ， 因 此 ， 若 maxHeap 
29 * 为 空 ， 则 minHeap 也 为 空 */ 

36 if (maxHeap.isEmpty()) { 

31 return ©; 

32 } 

33 if (maxHeap.size() == minHeap.size()) { 

34 return ((double)minHeap.peek()+(double)maxHeap.peek()) / 2; 
35 } else { 

36 /* 若 maxHeap 与 ninHeap 大 小 不 同 ， 那 么 ， 

37 maxHeap 必 定 多 一 个 元 素 ， 返 回 maxHeap 

38 * 的 堆 项 元 素 */ 

39 return maxHeap .peek(); 

46 } 

41 } 


18.10 ”给 定 两 个 字典 里 的 单词 ， 长 度 相等 。 编 写 一 个 方法 ， 将 一 个 单词 变换 成 另 一 个 单词 ， 
一 次 只 改动 一 个 字母 。 在 变换 过 程 中 ， 每 一 步 得 到 的 新 单词 都 必须 是 字典 里 存在 的 。( 第 106 页 ) 





解法 
此 题 看 似 困难 ,其实 只 要 将 广度 优先 搜索 稍 作 修改 就 可 以 解 出 来 。 在 “图 ”中 ,每 个 单词 的 3 
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所 有 分 文 ， 都 是 在 字典 里 相差 一 个 字母 的 单词 。 有 趣 的 地 方 在 于 实现 ,特别 是 ,我 们 能 一 边 实 现 


一 边 构建 这 张 图 吗 ? 
可 以 , 但 是 有 个 更 简单 的 方法 。 我 们 可 以 用 一 张 “ 回 溯 地 图 ”。 在 这 张 回溯 地 








图 中 , 如果 B[v] 





= w， 则 表示 编辑 v 可 得 到 w。 到 达 终 点 单词 时 ， 可 以 不 断 地 使 用 这 张 回溯 地 图 ， 往 回 找 出 路 径 。 














请 看 下 面 的 代码 : 


1 LinkedList<String> transform(String startWord, String stopWord, 
Set<String> dictionary) { 

3 startWord = startWord.toUpperCase(); 

4 stopWord = stopWord.toUpperCase(); 

5 Queuex<String> actionQueue = new LinkedList<String>(); 

6 Set<String> visitedSet = new HashSet<String>(); 

了 Map<String, String> backtrackMap = 
8 new TreeMap<String, String>(); 
9 


16 actionQueue.add(startNord ) ; 
1 visitedSet.add(startWord); 


12 

13 while (!actionQueue.isEmpty()) { 

14 String w = actionQueue.poll(); 

15 /* 对 每 个 变 成 W 只 需 编辑 一 次 的 单词 v */ 

16 for (String v : getOneEditWords(w)) { 
17 if (v.equals(stopWord)) { 

18 // 找到 我 们 的 单词 了 1 现在 往 回 走 
19 LinkedList<String> list = new LinkedList<String>(); 
26 // 将 v 追 加 至 1ist 

21 list.add(v); 

22 while (w != null) { 

23 list.add(@, Ww); 

24 w = backtrackMap.get(w); 

25 } 

26 return list; 

27 } 

28 /* 若 V 是 个 字典 里 的 单词 */ 

29 if (dictionary.contains(v)) { 

36 if (!visitedSet.contains(v)) { 
3 actionQueue.add(v); 

32 visitedSet.add(V); // 标记 为 已 访问 
33 backtrackMap.put(v, w); 

34 } 

35 } 

36 } 

37 } 

38 return null; 

39 } 

46 


41 Set<String> getOneEditWords(String word) { 
42 Set<String> words = new TreeSet<String>(); 
43 for (int i = 68; i < word.length(); i++) { 


44 char[] wordArray = word.toCharArray(); 
45 // 将 该 字母 改 成 别 的 字母 
46 for (char c = ‘A’; C<= 7’; c++) { 
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47 if (c != word.charAt(i)) { 

48 wordArray[i] = c; 

49 words.add(new String(wordArray)); 
56 } 

51 } 

52 } 

53 return words; 

54 } 


假设 n 为 初始 单词 的 长 度 , m 为 字典 里 相同 长 度 的 单词 个 数 。 其 中 while 循 环 最 多 会 拿 出 m 个 不 
同 的 单词 ， 故 此 算法 的 运行 时 间 为 OUz)。for 循 环 要 迭代 访问 整个 字符 串 ， 并 对 每 个 字符 施 以 
固定 次 数 的 蔡 换 操作 ， 时 间 复 杂 度 为 CCD。 


18.11 ”给 定 一 个 方 阵 ， 其 中 每 个 单元 (像素 ) 非 黑 即 白 。 设 计 一 个 算法 ， 找 出 四 条 边 皆 为 
黑色 像素 的 最 大 子 方 阵 。( 第 106 页 ) 


解法 
和 许多 问题 一 样 ， 此 题 也 有 难 易 两 种 解法 ， 下 面 将 逐一 讲解 。 


1.“ 简 单 ” 解 法 : O(N') 

我 们 知道 最 大 子 方 阵 的 长 度 可 能 为 N, 而 且 NxN 的 方 阵 只 有 一 个 , 很 容易 就 能 检查 这 个 方 阵 ， 
符合 要 求 则 返回 。 

如 果 找 不 到 NxN 的 方 阵 ， 可 以 尝试 第 二 大 的 子 方 阵 : (NWN-1) x (N-1)。 我 们 会 迭代 所 有 该 尺寸 
的 方 阵 , 一 旦 找到 符合 要 求 的 子 方 阵 ， 立 即 返 回 。 如 果 还 未 找到 ， 则 继续 尝试 N-2、N-3， 等 等 。 
由 于 我 们 是 从 大 到 小 逐 级 搜索 方 阵 ， 因 此 第 一 个 找到 的 必定 是 最 大 的 方 阵 。 

代码 具体 如 下 : 























1 Subsquare findSquare(int[][] matrix) { 

2 for (int i = matrix.length; i >= 1; i--) { 

3 Subsquare square = findSquareWithSize(matrix, i); 
4 if (square != null) return square; 
5 

6 

7 


return null; 
} 
8 
9 Subsquare findSquareWithSize(int[][] matrix, int squareSize) { 
16 /* 外 边 为 N 时 ， 里 头 会 有 (N - sz + 1) 个 边 长 
11 * 为 sz 的 方 阵 */ 


12 int count = matrix.length - squareSize + 1; 

13 

14 /* 迭代 所 有 边 长 为 sSquareSize 的 方 阵 */ 

15 for (int row = 6; row < count; row++) { 

16 for (int col = 6; col < count; col++) { 

17 if (isSquare(matrix, row, col, squareSize)) { 
18 return new Subsquare(row, col, squareSize); 
19 } 

26 } 

21 

22 return null; 
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25 boolean isSquare(int[][] matrix, int row, int col, int size) { 
26 // 检查 上 边界 和 下 边界 
27 for (int j = 86; j < size; j++){ 


28 if (matrix[row][col+j] == 1) { 

29 return false; 

30 

31 if (matrix[row+size-1][col+j] == 1){ 
32 Peturn false; 

33 } 

34 } 

35 

36 // 检查 左边 界 和 右边 界 

37 for (int i = 1; i < size - 1; i++){ 

38 if (matrix[row+i][col] == 1){ 

39 return false; 

40 } 

41 if (matrix[row+i][col+size-1] == 1){ 
42 return false; 

43 } 

44 } 

45 return true; 

46 } 


2. 预 处 理解 法 : O(N’) 

上 面 的 “简单 ”解法 之 所 以 执行 速度 慢 , 很 大 一 部 分 原因 在 于 ,每 次 检查 一 个 可 能 符合 要 求 
的 方 阵 ， 都 要 执行 ON) 的 工作 。 通 过 预先 做 些 处 理 ， 就 可 以 把 issquare 的 时 间 复 杂 度 降 为 0(1)， 
而 整个 算法 的 时 间 复 杂 度 降 至 OOV )。 

仔细 分 析 issquare 的 具体 用 处 ， 就 会 发 现 它 只 需 知道 特定 单元 下 方 及 右边 的 squaresize 项 
是 否 为 零 。 我 们 可 以 预先 以 直接 、 和 迭代 的 方式 算 好 这 些 数 据 。 

我 们 从 右 到 左 、 自 下 而 上 迭代 访问 每 个 单元 ， 并 执行 如 下 计算 :; 

if A[r][c] is white, zeros right and zeros below are 6 


else A[r][c].zerosRight = A[r][c + 1].zerosRight + 1 
A[r][c].zerosBelow = A[r + 1][c].zerosBelow + 1 












































下 面 这 个 例子 给 出 了 一 个 矩阵 的 相关 值 。 
(6@s right，6s below) Original Matrix 
80,0 | 1,3 | 9,6 W B W 
2,2 | 1,2 | 86,6 B B W 
2,1 | 1,1 | 8,6 B B W 





























现在 , issquare 方 法 不 必 再 迭代 O(N) 个 元 素 , 只 需 检 查 角落 的 zerosRight 和 zerosBelow 即 可 。 
下 面 是 该 算法 的 实现 代码 。 注 意 ，findsquare 和 findsquarewithsize 基 本 相同 ， 除 了 前 者 
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调用 了 processsquare" 以 及 之 后 操作 新 的 数据 类 型 。 





public class SquareCell { 

public int zerosRight = 6) 
public int zerosBelow = 0; 
/* 声明 、getter、setter */ 


Subsquare findSquare(int[][] matrix) { 
SquareCell[][] processed = processSquare(matrix); 


1 
2 
3 
4 
5 } 
6 
了 
8 
9 for (int i = matrix.length;j i >= 1; i--) { 


16 Subsquare square = findSquareWithSize(processed, i); 
11 if (square != null) return square; 

12 } 

13 return null; 

14 } 

15 


16 Subsquare findSquareWithSize(SquareCell[][] processed， 
17 int squareSize) { 
18 /* 与 第 一 个 算法 相同 */ 


19 } 

20 

21 

22 boolean isSquare(SquareCell[][] matrix, int row, int col, 
23 int size) { 


24 SquareCell topLeft = matrix[row][col]; 

25 SquareCell topRight = matrix[row][col + size - 1]; 
26 SquareCell bottomLeft = matrix[row + size - 1][coll]; 
27 if (topLeft.zerosRight < size) { // 检查 上 边界 


28 return false; 

29 } 

36 if (topLeft.zerosBelow < size) { // 检查 左边 界 
3 return false; 

32 

33 if (topRight.zerosBelow < size) { // 检查 右边 界 
34 return false; 

35 } 

36 if (bottomLeft.zerosRight < size) { // 检查 下 边界 
37 return false; 

38 } 

39 return true; 

46 } 

41 


42 SquareCell[][] processSquare(int[][] matrix) { 
43 SquareCell[][] processed = 


44 new SquareCell[matrix.length][matrix.1length]; 
45 

46 for (int r = matrix.length - 1; r >= 6j r--){ 
47 for (int c = matrix.length - 1; c >= 60; c--) { 
48 int rightZzeros = 0@; 

49 int belowZeros = 60; 








Q@ 原文 误 为 processMatrix。 一 一 译 者 注 
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56 // 只 有 单元 为 黑色 时 才 需 处 理 

51 if (matrix[r][c] == 6) { 

52 rightZzeros++; 

53 belowZerost++; 

54 // 下 一 列 在 同一 行 上 

55 if (c + 1 < matrix.length) { 

56 SquareCell previous = processed[r][c + 1]; 
57 rightZzeros += previous.zerosRight; 

58 } 

59 if (r + 1 < matrix.length) { 

66 SquareCell previous = processed[r + 1][c]; 
61 belowZeros += previous.zerosBelow; 

62 } 

63 

64 processed[r][c] = new SquareCell(rightZeros, belowZzeros); 
65 } 

66 } 

67 return processed; 

68 } 


18.12 ”给 定 一 个 正 整 数 和 负 整 数组 成 的 NxN 和 矩阵 ， 编 写 代码 找 出 元 素 总 和 最 大 的 子 和 矩 阵 。 
(第 106 页 ) 


解法 
此 题 有 很 多 种 解法 ,我 们 先 从 蛮 力 法 开始 ， 并 在 此 基础 上 进行 优化 。 


1. 变 力 法 : O(N) 

跟 许 多 “ 求 最 大 值 ”问题 一 样 ， 此 题 也 有 个 简单 的 蛮 力 解法 。 这 种 解法 就 是 直接 迄 代 所 有 可 
能 的 子 和 矩阵 ， 计 算 元 素 总 和 ， 找 出 最 大 值 。 

要 迭代 所 有 可 能 的 子 和 矩阵 ( 且 不 重复 )， 只 需 迭 代 所 有 的 有 序 行 配 对 ， 然 后 迭代 所 有 的 有 序 
列 配 对 。 








由 于 要 迭代 O(V9 个 子 和 矩阵， 计算 每 个 子 和 矩阵 的 元 素 总 和 用 时 O(V] ， 因 此 ， 这 个 解法 的 时 
间 复 杂 度 为 OOV9。 


2. 动态 规划 法 ，O(N ) 

注意 到 前 面 的 解法 被 拖 慢 了 OGVY)， 只 怪 和 矩阵 元 素 总 和 的 计算 太 慢 。 有 办 法 减少 元 素 总 和 计 
算 的 用 时 吗 ? 当然 有 ! 事实 上 ，computesum 的 用 时 可 以 降 至 O(1)。 

考虑 下 面 的 矩形 : 


Xx1 XxX2 





y1 








y2 
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假设 我 们 知道 下 面 这些 值 : 

ValD = area(point(86，6) -> point(x2, y2)) 
ValC = area(point(86，6) -> point(x2, y1)) 
ValB = area(point(86，6) -> point(x1, y2)) 
ValA = area(point(86，6) -> point(x1，y1)) 


每 个 Vval* 都 从 原点 开始 ， 在 子 矩 形 的 右 下 角 结 束 。 

利用 这 些 值 ， 可 得 到 以 下 等 式 : 

area(D) = ValD - area(A union C) - area(A union B) + area(A) 

或 者 ， 换 一 种 写法 : 

area(D) = ValD - ValB - ValC + ValA 

利用 类 似 的 逻辑 ， 就 可 以 有 效 地 为 矩阵 里 的 所 有 点 算出 这 些 值 : 

Val(x, y) = Val(x - 1, y) + Val(x, y - 1) - Val(x - 1, y - 1) + M[x][y] 
我 们 可 以 预先 算 好 这 些 值 ， 然 后 就 能 迅速 地 找到 元 素 总 和 最 大 的 子 和 矩阵 。 
下 面 是 该 算法 的 实现 代码 。 

















1 int getMaxMatrix(int[][] original) { 

2 int maxArea = Integer.MIN_VALUE; // 注意 , 最 大 总 和 可 能 小 于 8 
3 int rowCount = original.length; 

4 int columnCount = original[8].length; 

5 int[][] matrix = precomputeMatrix(original); 

6 for (int rowl = 6j rowl < rowCount; rowl++) { 

7 for (int row2 = rowl; row2 < rowCount; row2++) { 

8 for (int col1 = 6;j col1l < columnCount; col1++) { 

9 for (int col2 = col1; col2 < columnCount; col2++) { 
16 maxArea = Math.max(maxArea, computeSum(matrix, 
11 rowl, row2, coll, col2)); 

12 } 

13 } 

14 } 

15 } 

16 return maxArea; 

17 } 

18 


19 int[][] precomputeMatrix(int[][] matrix) { 
26 int[][] sumMatrix = new int[matrix.length][matrix[8].length]; 
21 for (int i = 6; i < matrix.length; i++) { 


22 for (int j = 8; j < matrix.length; j++) { 

23 if (i == 8 && j == 6) { // 第 一 个 单元 

24 sumMatrix[i][j] = matrix[i][j]; 

25 } else if (j == 6) { // 第 一 列 的 单元 

26 sumMatrix[i][j] = sumMatrix[i - 1][j] + matrix[i][j]; 
27 } else if (i == 68) { // 第 一 行 的 单元 

28 sumMatrix[i][j] = sumMatrix[i][j - 1] + matrix[i][j]; 
29 } else { 

36 sumMatrix[i][j] = sumMatrix[i - 1][j] + 

3 sumMatrix[i][j - 1] - sumMatrix[i - 1][j - 1] + 
32 matrix[i][j]; 

33 } 
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34 } 

35 } 

36 return sumMatrix; 
37 } 

38 


39 int computeSum(int[][] sumMatrix, int i1, int i2, int j1, int j2) { 
40 if (i1 == 6 && j1 == 6) { // 从 行 9、 列 8 开始 





41 return sumMatrix[i2][j2]; 

42 } else if (il == 6) { // 从 行 8 开始 

43 return sumMatrix[i2][j2] - sumMatrix[i2][j1 - 1]; 

44 } else if (jl == 6) { // 从 列 8 开 始 

45 return sumMatrix[i2][j2] - sumMatrix[il - 1][j2]; 

46 } else { 

47 return sumMatrix[i2][j2] - sumMatrix[i2][j1 - 1] 

48 - SumMatrix[il - 1][j2] + sumMatrix[il - 1][j1 - 1]; 
49 } 

50 } 


3. 优化 后 的 解法 ， O(N’) 
信和 不 信 由 你 ,但 确实 有 个 更 优 的 解法 。 如 果 和 矩 阵 为 R 行 C 列 ， 我 们 可 以 在 O(R*C) 时 间 内 解 出 





此 题 。 





回想 一 下 找 出 最 大 总 和 的 子 数组 问题 : 给 定 一 个 整数 数组 ， 找 出 元 素 总 和 最 大 的 子 数组 。 我 





们 有 办 法 在 ON) 时间 内 找到 (元素 总 和 ) 最 大 的 子 数组 ， 该 解法 也 可 用 来 求解 此 题 。 


么 








每 个 子 矩阵 都 可 以 表示 为 一 组 连续 的 行 和 一 组 连续 的 列 。 如 果 要 迭代 所 有 连续 行 的 组 合 , 那 
对 每 一 种 组 合 找 出 一 组 可 给 出 元 素 总 和 最 大 的 列 ， 就 可 以 了 。 也 就 是 说 : 


1 maxSum = 6 

2 foreach rowStart in rows 

3 foreach rowEnd in rows 

4 /* 我 们 有 一 些 子 矩 阵 ，rowStart 为 
5 # 矩阵 上 边 ，rowEnd 为 矩阵 下 边 ， 

6 * 找 出 colStart 和 colEnd 左 右 两 边 ， 
7 
8 
9 








* 使 得 总 和 最 大 */ 
maxSum = max(runningMaxSum, maxSum) 
return maxSum 





现在 ， 问 题 转变 为 如 何 高 效 地 找 出 “最 好 ”的 colstart 和 colEnd? 此 题 变 得 越 来 越 有 意思 了 。 
假设 有 如 下 子 和 矩阵 : 





























rowStart 
9 -8 1 3 -2 
-3 7 6 -2 4 
6 -4 -4 8 -7 
12 -5 3 9 -5 
rowEnd 
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我 们 想 要 找到 相应 的 colstart 和 colEnd， 使 得 rowstart 为 上 边 、rowEnd 为 下 边 的 子 矩 阵 元 
素 总 和 最 大 。 为 此 ， 我 们 可 以 把 每 一 列 加 起 来 ， 然 后 应 用 此 题 开 头 解 释 过 的 maxSubArray 函 数 。 

在 前 面 的 例子 中 , 总 和 最 大 的 子 数 组 是 第 1 列 到 第 4 列 。 这 就 意味 着 最 大 子 矩 阵 为 (rowStart， 
first column) 到 (rowEnd，fourth column)。 


至 此 ， 可 写 出 大 致 如 下 的 伪 码 。 


1 maxSum = 6 

2 foreach rowSstart in rows 

3 foreach rowEnd in rows 

4 foreach col in columns 

5 partialSum[col] = sum of matrix[rowStart, col] through 
6 matrix[rowEnd, col] 
4 
8 
9 





runningMaxSum = maxSubArray(partialSum) 
maxSum = max(runningMaxSum, maxSum) 
return maxSum 


第 5、6 行 计算 总 和 需 用 时 R*C ( 要 循环 访问 rowstart 至 rowEnd )， 因 此 全 部 用 时 O(R”C)。 不 
过 ， 大 功 尚未 告 成 。 

在 第 5、6 行 ， 人 从头 将 a[86]...a[i] 加 起 来 ， 即 使 在 外 层 for 循 环 的 前 一 次 迭代 时 已 计算 过 
a[6].. ,a[4_1] 的 总 和 。 这 这 部 分 重复 的 计算 完全 可 以 砍 掉 不 要 。 

1 maxSum = 6 

2 foreach rowStart in rows 

3 clear array partialsum 

foreach rowEnd in rows 


4 
5 foreach col in columns 

6 partialSum[col] += matrix[rowEnd, coll] 
了 

8 

9 














runningMaxSum = maxSubArray(partialSum) 
maxSum = max(runningMaxSum, maxSum) 
return maxSum 


最 终 ， 完 整 的 代码 大 致 如 下 : 


public void clearArray(int[] array) { 

for (int i = 6; i < array.length; i++) { 
array[i] = 8; 

} 


1 

2 

3 

4 

5 } 
6 

7 public static int maxSubMatrix(int[][] matrix) { 
8 int rowCount = matrix.length; 

9 int colCount = matrix[8].length; 

11 int[] partialSum = new int[colCount]; 

12 int maxSum = 6; // 最 大 总 和 是 个 空 矩 阵 


13 

14 for (int rowStart = 6;j rowStart < rowCount; rowStart++) { 

15 clearArray(partialSum); 

16 

17 for (int rowEnd = rowStart; rowEnd < rowCount; rowEnd++) { 
18 for (int i = 6;j i «< colCount; i++) { 
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19 partialSum[i] += matrix[rowEnd][i]; 
26 } 
21 
22 int tempMaxSum = maxSubArray(partialSsum, colCount); 
23 
24 /* 如 欲 追 踪 坐 标 ， 将 相关 代码 
25 * 放 在 这 里 */ 
26 maxSum = Math.max(maxSum, tempMaxSum); 
27 } 
28 } 
29 return maxSum; 
36 } 
31 
32 public static int maxSubArray(int array[], int N) { 
33 int maxSum = 0@; 
34 int runningSum = 0@; 
35 
36 for (int i = 6;j i < Ni i++) { 
37 runningSum += array[i]; 
38 maxSum = Math.max(maxSum, runningSum); 
39 
40 /* 若 runningSum < 6， 就 没 必要 再 继续 了 。 
41 * 重 置 */ 
42 if (runningSum < 6) { 
43 runningSum = 6) 
44 } 
45 } 
46 return maxSum; 
47 } 





此 题 非 常 复杂 ， 若 没有 面试 官 的 大 量 提示 和 帮助 ， 在 面试 中 很 难 周全 地 解 出 整个 问题 。 


18.13 ”给 定 一 份 几 百 万 个 单词 的 清单 ， 设 计 一 个 算法 ， 创 建 由 字母 组 成 的 最 大 矩形 ， 其 中 
每 一 行 组 成 一 个 单词 ( 自 左 向 右 ), 每 一 列 也 组 成 一 个 单词 ( 自 上 而 下 )。 不 要 求 这 些 单词 在 清单 
里 连续 出 现 ， 但 要 求 所 有 行 等 长 ， 所 有 列 等 高 。( 第 106 页 ) 


解法 


很 多 与 字典 有 关 的 问题 ， 通 过 预先 做 些 处 理 就 可 以 解 出 来 。 对 于 此 题 ， 哪 一 部 分 可 以 做 预 


理 呢 ? 
好 n 








I 
Bn 











处 


， 如 果 要 创建 一 个 单词 矩形 ， 就 必须 满足 以 下 要 求 : 每 一 行 等 长 ， 每 一 列 等 高 。 因 此 ,我 














们 可 以 将 这 个 字典 的 单词 按 长 短 进行 分 组 , 姑且 把 这 个 分 组 叫 作 D 
接 下 来 ， 观 察 要 找 的 最 大 矩形 。 可 能 形成 的 绝对 最 大 的 矩形 有 多 大 呢 ? 它 会 是 length 
(largest word)2。 


1 


2 
3 
4 
5 
6 


int maxRectangle = longestWord * longestWord; 
for z = maxRectangle to 1 { 
for each pair of numbers (i, j) where i*j =z{ 


} 


} 


/* 试 着 用 单词 构建 矩形 ， 成 功 则 返回 */ 
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中 D[i] 包 含 长 度 为 的 单词 
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从 最 大 可 能 的 抢 形 迭代 至 最 小 的 矩形 , 可 以 保证 第 一 个 找到 的 符合 要 求 的 矩形 就 是 题目 要 求 
的 最 大 矩形。 

现在 ， 轮 到 困难 的 部 分 : makeRectangle(int 1，int h)。 这 个 方法 试图 构建 长 /高 4 的 单词 
矩形。 

一 种 做 法 是 迭代 所 有 长 的 有 序 单词 集合 ， 然 后 检查 每 一 列 字母 是 否 形成 有 效 单词 。 这 么 做 
也 行 得 通 ， 但 是 非常 低 效 。 

假设 我 们 正 试 着 构造 6x$ 的 矩形， 前 几 行 单词 如 下 : 




















至 此 可 知 , 第 一 列 开头 几 个 字母 为 tqp。 我 们 知道 或 者 说 应 该 知道 , 字典 里 没有 以 tqp 开 头 的 
单词 。 既 然 明摆着 最 终 创建 不 出 有 效 的 矩形， 为 何 还 要 自 寻 烦恼 ， 继续 构造 下 去 呢 ? 

这 就 引出 一 个 更 优 的 解法 。 我 们 可 以 构建 一 棵 单词 查找 树 (trie )， 从 而 轻易 查 出 某 个 子 串 是 
否 为 字典 里 单词 的 前 级。 然后 , 在 一 行 一 行 自 上 而 下 构造 矩形 时 ,检查 每 一 列 字母 是 否 均 为 有 效 
前 缀 。 如 果 不 是 ， 则 立即 失败 并 中 止 ， 不 再 继续 构造 这 个 矩形 。 

下 面 是 该 算法 的 实现 代码 ， 长 且 复 全， 下 面 我 们 会 逐步 解说 。 

一 开始 会 做 些 预 处 理 ， 将 单词 按 长 度 分 组 。 我 们 会 创建 一 个 单词 查找 树 〈 每 一 个 trie 包 含 某 
长 度 的 单词 ) 数组 ， 但 直到 真正 需要 时 ， 才 会 构建 单词 查找 树 。 


1 WordGroup[] groupList = WordGroup.createWordGroups(1ist); 
2 int maxWordLength = groupList.length; 
3 Trie trieList[] = new Trie[maxWordLength]; 


maxRectangle 方 法 是 代码 的 “主体 ”"”， 从 可 能 的 最 大 矩形 ( maxWordLength2 ) 开始 ， 然 后 试 
着 构建 该 大 小 的 矩形 。 车 构建 失败 , 该 方法 会 将 最 大 面积 减 一 , 并 尝试 新 的 、 较 小 的 尺寸 。 由 此 ， 
第 一 个 成 功 构 建 的 矩形 必定 是 最 大 的 。 
























































1 Rectangle maxRectangle() { 

2 int maxSize = maxWordLength * maxWordLength; 

3 for (int z = maxSize; z > 60; z--) { // 从 最 大 面积 开始 
4 for (int i = 1; i <= maxWordLength; i++) { 

5 if (z % i == 6) { 

6 int j] =z/ i; 

7 if (j <= maxWordLength) { 

8 /* 构造 长 度 T、 高 度 j 的 和 矩形 。 注 意 ， 

9 半 工 *j = Z#/ 

16 Rectangle rectangle = makeRectangle(i, j); 
11 if (rectangle != null) { 

12 return rectangle; 


18 return null; 
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maxRectangle 又 调用 了 makeRectangle 方 法 ， 用 于 构造 指定 长 度 和 高 度 的 矩形 。 


1 
这 
3 
4 
5 
6 
7 
8 


9 
16 
dd 
1 
13 
14 
15 } 


Rectangle makeRectangle(int length, int height) { 


if (groupList[length-1] == null || 
groupList[height-1] == null) { 
return null; 


} 


/* 若 不 存在 ， 就 构建 该 单词 长 度 的 trie */ 

if (trieList[height - 1] == null) { 
LinkedList<String> words = groupList[height - 1].getWords(); 
trieList[height - 1] = new Trie(words); 


} 


return makePartialRectangle(length, height, 
new Rectangle(length)); 


makePartialRectangle 方 法 真正 负责 构建 算 形 ， 参 数 为 预期 的 最 终 长 度 和 高 度 以 及 部 分 成 

















形 的 和 矩形。 如果 和 拖 形 的 高 度 已 达到 最 后 想 要 的 高 度 ， 就 直接 查看 每 一 列 能 和 否 构 成 有 效 、 完 整 的 单 
词 ， 然 后 返回 。 
否则 ,检查 每 一 列 字 母 能 否 构成 有 效 前 缀 。 如 若 不 能 ， 就 立即 中 止 ， 因 为 这 个 部 分 成 形 的 矩 


形 最 后 不 可 能 构建 出 有 效 的 矩形 。 
不 过 ， 如 果 到 目前 为 止 一 切 顺利 ， ee 那么 ， 就 继续 搜索 相应 长 度 
追加 至 当前 矩形 的 后 面 ， 然 后 进入 递归 试 着 以 { 追 加 中 新 单词 的 矩形 } 为 基础 构建 矩形 。 


的 单词 ， 











Rectangle makePartialRectangle(int 1, int h, Rectangle rectangle) { 


if (rectangle.height == h) { // 检查 移 形 是 否 已 完成 
if (rectangle.isComplete(1，h，SgroupList[h - 1])) { 
return rectangle; 
} else { 
return null; 
} 
} 


/* 将 所 有 列 与 trie 比 较 ， 检 查 是 否 有 效 */ 
if (!rectangle.ispartialOK(1, trieList[h - 1])) { 
return null; 


} 


/* 迁 代 访问 该 长 度 的 所 有 单词 ， 并 加 入 
* 当前 的 部 分 矩形 ， 然 后 试 着 递归 构建 出 
* 和 矩形 */ 
for (int i = 6; i < groupList[1-1].length(); i++) { 
/* 当前 矩形 加 上 新 单词 构建 新 和 矩形 */ 
Rectangle orgPlus = 
rectangle.append(groupList[1-1].getWord(i)); 


/* 试 着 以 这 个 新 的 、 部 分 矩形 构建 新 矩形 */ 


Rectangle rect = makePartialRectangle(1, h, orgPlus); 
if (rect != null) { 
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26 return rect; 
27 } 

28 } 

29 return null; 

30 } 





Rectangle 类 代表 一 个 部 分 或 完整 的 单词 矩形 ， 可 以 调用 方法 isPartial0k 来 检查 矩形 到 有 目 
前 为 止 是 否 有 效 ( 即 每 一 列 都 是 有 效 的 单词 前 级 )。 方 法 1sComplete 的 功能 类 似 ， 不 过 只 检查 每 
一 列 是 否 为 完整 的 单词 。 
public class Rectangle { 


public int height, length; 
public char [][] matrix; 


1 
之 
3 
4 
5 /* 构造 一 个 “ 空 ” 的 和 矩形， 长 度 是 固定 的 ， 
6 * 但 高 度 会 随 着 单词 的 加 入 而 变化 */ 
7 public Rectangle(int 1) { 

8 height = 9; 

9 length = 1; 


10  } 


12 /* 根据 指定 长 度 和 高 度 的 字符 数组 


13 * 构造 矩形 ， 使 用 指定 的 字母 矩阵 

14 *# 表示 (假定 参数 指定 的 长 度 和 高 

15 * 度 与 数组 参数 的 大 小 

16 * 相符 ) */ 

17 public Rectangle(int length, int height, char[][] letters) { 
18 this.height = letters.length; 

19 this.length = letters[8].length; 

26 matrix = letters; 

21 } 

22 

23 public char getLetter (int i, int j) { return matrix[i][j]; } 
24 public String getColumn(int i) { ... } 

25 


26 /* 检查 所 有 列 是 否 都 为 有 效 。 所 有 列 已 知 为 
27 * 有 效 的 ， 因 为 它们 是 直接 从 字典 里 取出 的 */ 
28 public boolean isComplete(int 1, int h, WordGroup groupList) { 


29 if (height == h) { 

36 /* 检查 每 一 列 是 否 为 字典 里 的 单词 */ 

31 for (int i = 6;j i < 1; i++) { 

32 String col = getColumn(i); 

33 if (!groupList.containsWord(col)) { 
34 return false; 

35 } 

36 } 

37 return true; 

38 } 

39 return false; 

46 } 

41 

42 public boolean ispartialOK(int 1, Trie trie) { 
43 if (height == 6) return true; 
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44 for (int i = 6j i < 1; i++ ) { 
45 String col = getColumn(i); 
46 if (!trie.contains(col)) { 
47 return false; 

48 } 

49 } 

56 return true; 

51 } 

52 

53 /* 在 当前 矩形 上 追加 s 来 新 建 

54 * Rectangle */ 

55 public Rectangle append(String s) { ... } 
56 } 

















WordGroup 类 是 个 简单 的 容 缮 ， 包 含 某 长 度 的 所 有 单词 。 为 方便 查找 ， 我 们 会 将 单词 储存 在 
散 列 表 和 ArrayList 中 。 
WordGroup 中 的 列表 由 静态 方法 createWordGroups 创 建 。 


1 public class WordGroup { 

2 private Hashtable<String, Boolean> lookup = 
3 new Hashtable<String, Boolean>(); 

4 private ArrayList<String> group = new ArrayList<String>(); 
5 

6 public boolean containsWord(String s) { 

7 return lookup.containsKey(s); 

8 } 

9 

16 public void addWord (String s) { 

11 group.add(s); 

12 lookup.put(s, true); 

13 } 

14 


5 public int length() { return group.size(); } 
16 public String getWord(int i) { return group.get(i); } 
17 public ArrayList<String> getWords() { return group; } 


18 

19 public static WordGroup[] createWordGroups(String[] list) { 
26 WordGroup[] groupList; 

21 int maxWordLength = 0@; 

22 /* 找 出 最 长 单词 的 长 度 */ 

23 for (int i = 6; i < list.length; i++) { 

24 if (list[i].length() > maxWordLength) { 
25 maxWordLength = list[i].length(); 

26 } 

27 } 

28 

29 /* 将 字典 里 的 单词 按 长 度 分 组 ， 相 同 长 度 的 分 为 一 组 。 
36 * groupList[i] 会 包含 一 串 单 词 ， 每 个 单词 的 长 度 为 
31 * length (i+1) */ 

32 groupList = new WordGroup[maxWordLength]; 

33 for (int i = 6; i < list.length; i++) { 

34 /* 这 里 是 wordLength - 1 而 非 wordLength， 

35 * 因为 这 里 是 索引 ， 不 存在 长 度 为 8 的 单词 */ 
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36 int wordLength = list[i].length() - 1; 

37 if (groupList[wordLength] == null) { 

38 groupList[wordLength] = new WordGroup(); 
39 

46 groupList[wordLength].addWword(1list[i]); 

41 } 

42 return groupList; 

43 } 

44 } 


此 题 完整 代码 ( 包括 Trie 和 TrieNode 的 )， 可 在 本 书 所 附 的 源码 包 里 找 出 。 注 意 ， 面 对 复杂 
如 是 的 问题 , 你 很 可 能 只 需要 写 出 伪 码 即 可 。 毕 竞 , 要 在 这 么 短 的 时 间 内 写 出 全 部 代码 几乎 是 不 
可 能 的 。 
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“如 果 你 正 打 算 参 加 技术 面试 ， 我 极力 推荐 你 阅读 
此 书 。 这 本 书 汇总 了 诸多 你 不 可 不 知 的 决胜 于 技术 
面试 的 问题 、 策 略 和 方法 。” 


一 一 Ginnie ， 亚 马 逊 评论 者 


> 150 个 编程 题 问 答 


从 二 叉 树 到 二 分 查找 ， 该 部 分 涵盖 了 关于 数据 结构 和 算法 的 最 
常见 、 最 有 用 的 面试 题 以 及 最 为 精巧 的 解决 方案 。 


> ”应 对 棘手 算法 题 的 5 种 行 之 有 效 的 方法 


通过 这 5 种 方法 ， 你 可 以 学 会 如 何 处 理 并 攻克 算法 难题 ， 包 括 
那些 最 棘手 的 算法 题 。 


> ”面试 者 最 容易 犯 的 10 个 错误 


不 要 因为 这 些 常见 的 错误 而 与 成 功 失之交臂 。 要 了 解 面 试 者 常 
犯 的 一 些 错 误 ， 学 会 如 何 避 免 这 些 问题 。 


> ”面试 准备 的 若干 策略 


不 要 因为 沉溺 在 无 穷 无 尽 的 面试 题 中 而 错过 了 最 重要 的 求职 建 
议 。 这 些 策略 和 步骤 可 以 让 你 更 有 效 地 准备 面试 。 
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三 
图 灵 和 信 区 
最 前 沿 的 [T 类 电子 书 发 售 平台 


有 子 出 版 的 时 代 已 经 来 临 。 在 许多 出 版 界 同 行 还 在 犹 图 灵 社 区 进一步 把 传统 出 版 流程 与 电子 书 出 版 业务 
驳 丛 得 的 时 候 ， 图 灵 社 区 已 经 采取 实际 行动 拥抱 这 个 紧密 结合 ， 目 前 已 实现 作 译 者 网 上 交 稿 、 编 辑 网 上 
出 版 业 巨变 。 作 为 国内 第 一 家 发 售 电子 图 书 的 IT 类 出 审 稿 、 按 章 发 布 的 电子 出 版 模式 。 这 种 新 的 出 版 模 
版 商 ， 图 灵 社 区 目前 为 读者 提供 两 种 DRM-free 的 阅读 式 ， 我 们 称 之 为 “敏捷 出 版 ”， 它 可 以 让 读者 以 较 
体验 : 在 线 阅读 和 PDF。 快 的 速度 了 解 到 国外 最 新 技术 图 书 的 内 容 ， 弥 补 以 
往 翻 译 版 技术 书 “ 出 版 即 过 时 ”的 缺憾 。 同 时 ， 敏 
相 比 纸 质 书 ， 电 子 书 具 有 许多 明显 的 优势 。 它 不 仅 发 捷 出 版 使 得 作 、 译 、 编 、 读 的 交流 更 为 方便 ， 可 以 
布 快 ， 更 新 容易 ， 而 且 尽 可 能 采用 了 彩色 图 片 (即使 AT A ee 
有 的 书 纸 质 版 是 黑白 印刷 的 ) 。 读 者 还 可 以 方便 地 进 
搜索、 剪贴、 复制 和 打印 。 




























































































































































































































































































一 人 


最 方便 的 开放 出 版 平台 最 直接 的 读者 交流 平台 


图 灵 社 区 向 读者 开放 在 线 写作 功能 ， 协 助 你 实现 自 出 在 图 灵 社 区 ， 你 可 以 十 分 方便 地 写作 文章 、 提 交 勘 









































































































































































































































版 和 开源 出 版 的 梦想 。 利 用 “合集 ”功能 ， 你 就 能 联 误 、 发 表 评 论 ， 以 各 种 方式 与 作 译 者 、 编 辑 人 员 和 
合 二 三 好 友 共 同 创作 一 部 技术 参考 书 ， 以 免费 或 收费 其 他 读者 进行 交流 互动 。 提 交 勘 误 还 能 够 获 赠 社区 
的 形式 提供 给 读者 。 (收费 形式 须 经 过 图 灵 社 区 立项 银子 。 

评审 。) 这 极 大 地 降低 了 出 版 的 门槛 。 只 要 你 有 写作 

的 意愿 ， 图 灵 社 区 就 能 帮助 你 实现 这 个 梦想 。 成 熟 的 你 可 以 积极 参与 社区 经 常 开展 的 访谈 、 审 读 、 评 选 
书稿 ， 有 机 会 入 选 出 版 计划 ， 同 时 出 版 纸 质 书 。 等 多 种 活动 ， 赢 取 积分 和 银子 ， 积 累 个 人 声望 。 
图 灵 社 区 引进 出 版 的 外 文 图 书 ， 都 将 在 立项 后 马上 在 

社区 公布 。 如 果 你 有 意 翻译 哪 本 图 书 ， 欢 迎 你 来 社区 
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请 。 只 要 你 通过 试 译 的 考验 ， 即 可 签约 成 为 图 灵 的 
译 者 。 当 然 ， 要 想 成 功 地 完成 一 本 书 的 翻译 工作 ， 是 
需要 有 坚强 的 新 力 的 。 
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